@goscribe/server 1.3.4 → 1.6.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 (383) 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/context.d.ts +14 -1
  6. package/dist/context.js +23 -2
  7. package/dist/controllers/admin.controller.d.ts +715 -0
  8. package/dist/controllers/admin.controller.js +9 -0
  9. package/dist/controllers/annotations.controller.d.ts +439 -0
  10. package/dist/controllers/annotations.controller.js +9 -0
  11. package/dist/controllers/app-router.controller.d.ts +3011 -0
  12. package/dist/controllers/app-router.controller.js +38 -0
  13. package/dist/controllers/app-router.controller.test.d.ts +1 -0
  14. package/dist/controllers/app-router.controller.test.js +36 -0
  15. package/dist/controllers/auth.controller.d.ts +323 -0
  16. package/dist/controllers/auth.controller.js +9 -0
  17. package/dist/controllers/base.controller.d.ts +4 -0
  18. package/dist/controllers/base.controller.js +5 -0
  19. package/dist/controllers/chat.controller.d.ts +341 -0
  20. package/dist/controllers/chat.controller.js +9 -0
  21. package/dist/controllers/copilot.controller.d.ts +397 -0
  22. package/dist/controllers/copilot.controller.js +9 -0
  23. package/dist/controllers/flashcards.controller.d.ts +651 -0
  24. package/dist/controllers/flashcards.controller.js +9 -0
  25. package/dist/controllers/members.controller.d.ts +339 -0
  26. package/dist/controllers/members.controller.js +9 -0
  27. package/dist/controllers/notifications.controller.d.ts +199 -0
  28. package/dist/controllers/notifications.controller.js +9 -0
  29. package/dist/controllers/payment.controller.d.ts +181 -0
  30. package/dist/controllers/payment.controller.js +9 -0
  31. package/dist/controllers/podcast.controller.d.ts +575 -0
  32. package/dist/controllers/podcast.controller.js +9 -0
  33. package/dist/controllers/router-module.controller.d.ts +5 -0
  34. package/dist/controllers/router-module.controller.js +6 -0
  35. package/dist/controllers/studyguide.controller.d.ts +73 -0
  36. package/dist/controllers/studyguide.controller.js +9 -0
  37. package/dist/controllers/worksheets.controller.d.ts +829 -0
  38. package/dist/controllers/worksheets.controller.js +9 -0
  39. package/dist/controllers/workspace.controller.d.ts +1207 -0
  40. package/dist/controllers/workspace.controller.js +9 -0
  41. package/dist/lib/activity_human_description.test.js +16 -15
  42. package/dist/lib/activity_log_service.test.js +28 -23
  43. package/dist/lib/ai/config.d.ts +20 -0
  44. package/dist/lib/ai/config.js +31 -0
  45. package/dist/lib/ai/embedding-client.d.ts +8 -0
  46. package/dist/lib/ai/embedding-client.js +30 -0
  47. package/dist/lib/ai/index.d.ts +48 -0
  48. package/dist/lib/ai/index.js +29 -0
  49. package/dist/lib/ai/inference-backend/client.d.ts +28 -0
  50. package/dist/lib/ai/inference-backend/client.js +301 -0
  51. package/dist/lib/ai/inference-backend/mocks.d.ts +12 -0
  52. package/dist/lib/ai/inference-backend/mocks.js +133 -0
  53. package/dist/lib/ai/inference-backend/types.d.ts +44 -0
  54. package/dist/lib/ai/inference-backend/types.js +1 -0
  55. package/dist/lib/ai/json-parse.d.ts +2 -0
  56. package/dist/lib/ai/json-parse.js +34 -0
  57. package/dist/lib/ai/llm-client.d.ts +7 -0
  58. package/dist/lib/ai/llm-client.js +36 -0
  59. package/dist/lib/ai/mock.d.ts +2 -0
  60. package/dist/lib/ai/mock.js +10 -0
  61. package/dist/lib/ai/types.d.ts +9 -0
  62. package/dist/lib/ai/types.js +1 -0
  63. package/dist/lib/chunking.d.ts +19 -0
  64. package/dist/lib/chunking.js +47 -0
  65. package/dist/lib/curated-kb-seed.d.ts +12 -0
  66. package/dist/lib/curated-kb-seed.js +155 -0
  67. package/dist/lib/email.js +67 -108
  68. package/dist/lib/embeddings.d.ts +2 -0
  69. package/dist/lib/embeddings.js +1 -0
  70. package/dist/lib/ensure-curated-kb-catalog.d.ts +6 -0
  71. package/dist/lib/ensure-curated-kb-catalog.js +53 -0
  72. package/dist/lib/env.d.ts +1 -5
  73. package/dist/lib/env.js +2 -7
  74. package/dist/lib/inference.d.ts +1 -8
  75. package/dist/lib/inference.js +1 -19
  76. package/dist/lib/kb-meta.d.ts +8 -0
  77. package/dist/lib/kb-meta.js +77 -0
  78. package/dist/lib/note-text.d.ts +1 -0
  79. package/dist/lib/note-text.js +47 -0
  80. package/dist/lib/notification-service.test.js +37 -36
  81. package/dist/lib/pdf.d.ts +11 -0
  82. package/dist/lib/pdf.js +11 -0
  83. package/dist/lib/usage_service.d.ts +2 -1
  84. package/dist/lib/usage_service.js +30 -12
  85. package/dist/lib/worksheet-generation.js +4 -4
  86. package/dist/lib/worksheet-generation.test.js +32 -17
  87. package/dist/lib/workspace-kb.d.ts +5 -0
  88. package/dist/lib/workspace-kb.js +7 -0
  89. package/dist/models/controller-context.model.d.ts +8 -0
  90. package/dist/models/controller-context.model.js +1 -0
  91. package/dist/repositories/artifact.repository.d.ts +60 -0
  92. package/dist/repositories/artifact.repository.js +40 -0
  93. package/dist/repositories/base.repository.d.ts +14 -0
  94. package/dist/repositories/base.repository.js +14 -0
  95. package/dist/repositories/invitation.repository.d.ts +94 -0
  96. package/dist/repositories/invitation.repository.js +44 -0
  97. package/dist/repositories/notification.repository.d.ts +72 -0
  98. package/dist/repositories/notification.repository.js +44 -0
  99. package/dist/repositories/router-module.repository.d.ts +10 -0
  100. package/dist/repositories/router-module.repository.js +14 -0
  101. package/dist/repositories/user.repository.d.ts +74 -0
  102. package/dist/repositories/user.repository.js +37 -0
  103. package/dist/repositories/workspace-member.repository.d.ts +31 -0
  104. package/dist/repositories/workspace-member.repository.js +31 -0
  105. package/dist/repositories/workspace.repository.d.ts +97 -0
  106. package/dist/repositories/workspace.repository.js +79 -0
  107. package/dist/routers/_app.d.ts +566 -111
  108. package/dist/routers/_app.js +4 -0
  109. package/dist/routers/admin.d.ts +0 -4
  110. package/dist/routers/admin.js +21 -549
  111. package/dist/routers/annotations.js +12 -170
  112. package/dist/routers/artifactVersions.d.ts +65 -0
  113. package/dist/routers/artifactVersions.js +14 -0
  114. package/dist/routers/auth.d.ts +0 -6
  115. package/dist/routers/auth.js +37 -422
  116. package/dist/routers/chat.js +15 -229
  117. package/dist/routers/copilot.d.ts +14 -13
  118. package/dist/routers/copilot.js +13 -532
  119. package/dist/routers/flashcards.d.ts +17 -6
  120. package/dist/routers/flashcards.js +23 -349
  121. package/dist/routers/knowledgeBase.d.ts +421 -0
  122. package/dist/routers/knowledgeBase.js +118 -0
  123. package/dist/routers/members.d.ts +0 -41
  124. package/dist/routers/members.js +22 -710
  125. package/dist/routers/notes.d.ts +94 -0
  126. package/dist/routers/notes.js +37 -0
  127. package/dist/routers/notifications.js +7 -109
  128. package/dist/routers/payment.d.ts +2 -12
  129. package/dist/routers/payment.js +11 -393
  130. package/dist/routers/podcast.d.ts +1 -1
  131. package/dist/routers/podcast.js +11 -784
  132. package/dist/routers/studyguide.js +3 -129
  133. package/dist/routers/worksheets.d.ts +29 -14
  134. package/dist/routers/worksheets.js +49 -628
  135. package/dist/routers/workspace.d.ts +27 -71
  136. package/dist/routers/workspace.js +29 -922
  137. package/dist/scripts/purge-deleted-users.js +2 -2
  138. package/dist/server.js +10 -3
  139. package/dist/services/activity/activity-human-description.service.d.ts +13 -0
  140. package/dist/services/activity/activity-human-description.service.js +221 -0
  141. package/dist/services/activity/activity-human-description.service.test.d.ts +1 -0
  142. package/dist/services/activity/activity-human-description.service.test.js +16 -0
  143. package/dist/services/activity/activity-log.service.d.ts +87 -0
  144. package/dist/services/activity/activity-log.service.js +276 -0
  145. package/dist/services/activity/activity-log.service.test.d.ts +1 -0
  146. package/dist/services/activity/activity-log.service.test.js +27 -0
  147. package/dist/services/activity-human-description.service.d.ts +13 -0
  148. package/dist/services/activity-human-description.service.js +221 -0
  149. package/dist/services/activity-human-description.service.test.d.ts +1 -0
  150. package/dist/services/activity-human-description.service.test.js +16 -0
  151. package/dist/services/activity-log.service.d.ts +87 -0
  152. package/dist/services/activity-log.service.js +276 -0
  153. package/dist/services/activity-log.service.test.d.ts +1 -0
  154. package/dist/services/activity-log.service.test.js +27 -0
  155. package/dist/services/admin/admin.service.d.ts +270 -0
  156. package/dist/services/admin/admin.service.js +476 -0
  157. package/dist/services/admin.service.d.ts +270 -0
  158. package/dist/services/admin.service.js +476 -0
  159. package/dist/services/ai/ai-session.service.d.ts +5 -0
  160. package/dist/services/ai/ai-session.service.js +4 -0
  161. package/dist/services/ai-session.service.d.ts +60 -0
  162. package/dist/services/ai-session.service.js +561 -0
  163. package/dist/services/annotation.service.d.ts +177 -0
  164. package/dist/services/annotation.service.js +154 -0
  165. package/dist/services/artifact-notification.service.d.ts +14 -0
  166. package/dist/services/artifact-notification.service.js +20 -0
  167. package/dist/services/artifact-version.service.d.ts +38 -0
  168. package/dist/services/artifact-version.service.js +129 -0
  169. package/dist/services/artifacts/annotation.service.d.ts +177 -0
  170. package/dist/services/artifacts/annotation.service.js +154 -0
  171. package/dist/services/artifacts/artifact-version.service.d.ts +38 -0
  172. package/dist/services/artifacts/artifact-version.service.js +129 -0
  173. package/dist/services/artifacts/chat.service.d.ts +127 -0
  174. package/dist/services/artifacts/chat.service.js +182 -0
  175. package/dist/services/artifacts/study-guide.service.d.ts +18 -0
  176. package/dist/services/artifacts/study-guide.service.js +65 -0
  177. package/dist/services/auth/auth.service.d.ts +94 -0
  178. package/dist/services/auth/auth.service.js +368 -0
  179. package/dist/services/auth.service.d.ts +94 -0
  180. package/dist/services/auth.service.js +368 -0
  181. package/dist/services/base.service.d.ts +14 -0
  182. package/dist/services/base.service.js +14 -0
  183. package/dist/services/billing/payment.service.d.ts +44 -0
  184. package/dist/services/billing/payment.service.js +365 -0
  185. package/dist/services/billing/subscription.service.d.ts +37 -0
  186. package/dist/services/billing/subscription.service.js +654 -0
  187. package/dist/services/billing/usage.service.d.ts +47 -0
  188. package/dist/services/billing/usage.service.js +149 -0
  189. package/dist/services/chat.service.d.ts +127 -0
  190. package/dist/services/chat.service.js +182 -0
  191. package/dist/services/content/copilot.service.d.ts +113 -0
  192. package/dist/services/content/copilot.service.js +439 -0
  193. package/dist/services/content/flashcard-progress.service.d.ts +159 -0
  194. package/dist/services/content/flashcard-progress.service.js +432 -0
  195. package/dist/services/content/flashcard.service.d.ts +184 -0
  196. package/dist/services/content/flashcard.service.js +339 -0
  197. package/dist/services/content/media-analysis.service.d.ts +23 -0
  198. package/dist/services/content/media-analysis.service.js +404 -0
  199. package/dist/services/content/podcast.service.d.ts +267 -0
  200. package/dist/services/content/podcast.service.js +653 -0
  201. package/dist/services/content/worksheet-content.service.d.ts +37 -0
  202. package/dist/services/content/worksheet-content.service.js +84 -0
  203. package/dist/services/content/worksheet-content.service.test.d.ts +1 -0
  204. package/dist/services/content/worksheet-content.service.test.js +69 -0
  205. package/dist/services/content/worksheet-generation.service.d.ts +91 -0
  206. package/dist/services/content/worksheet-generation.service.js +95 -0
  207. package/dist/services/content/worksheet-generation.service.test.d.ts +1 -0
  208. package/dist/services/content/worksheet-generation.service.test.js +20 -0
  209. package/dist/services/content/worksheet.service.d.ts +347 -0
  210. package/dist/services/content/worksheet.service.js +599 -0
  211. package/dist/services/copilot.service.d.ts +116 -0
  212. package/dist/services/copilot.service.js +447 -0
  213. package/dist/services/flashcard-progress.service.d.ts +2 -2
  214. package/dist/services/flashcard-progress.service.js +3 -2
  215. package/dist/services/flashcard.service.d.ts +140 -0
  216. package/dist/services/flashcard.service.js +325 -0
  217. package/dist/services/invitation.service.d.ts +66 -0
  218. package/dist/services/invitation.service.js +348 -0
  219. package/dist/services/knowledge/knowledge-base.service.d.ts +316 -0
  220. package/dist/services/knowledge/knowledge-base.service.js +544 -0
  221. package/dist/services/knowledge-base.service.d.ts +316 -0
  222. package/dist/services/knowledge-base.service.js +536 -0
  223. package/dist/services/media-analysis.service.d.ts +23 -0
  224. package/dist/services/media-analysis.service.js +384 -0
  225. package/dist/services/member.service.d.ts +36 -0
  226. package/dist/services/member.service.js +193 -0
  227. package/dist/services/members/invitation.service.d.ts +66 -0
  228. package/dist/services/members/invitation.service.js +348 -0
  229. package/dist/services/members/member.service.d.ts +36 -0
  230. package/dist/services/members/member.service.js +193 -0
  231. package/dist/services/note.service.d.ts +55 -0
  232. package/dist/services/note.service.js +111 -0
  233. package/dist/services/notification.service.d.ts +214 -0
  234. package/dist/services/notification.service.js +550 -0
  235. package/dist/services/notification.service.test.d.ts +1 -0
  236. package/dist/services/notification.service.test.js +87 -0
  237. package/dist/services/notifications/notification.service.d.ts +214 -0
  238. package/dist/services/notifications/notification.service.js +550 -0
  239. package/dist/services/notifications/notification.service.test.d.ts +1 -0
  240. package/dist/services/notifications/notification.service.test.js +87 -0
  241. package/dist/services/payment.service.d.ts +55 -0
  242. package/dist/services/payment.service.js +368 -0
  243. package/dist/services/podcast.service.d.ts +267 -0
  244. package/dist/services/podcast.service.js +654 -0
  245. package/dist/services/router-module.service.d.ts +7 -0
  246. package/dist/services/router-module.service.js +10 -0
  247. package/dist/services/study-guide.service.d.ts +18 -0
  248. package/dist/services/study-guide.service.js +65 -0
  249. package/dist/services/subscription.service.d.ts +37 -0
  250. package/dist/services/subscription.service.js +654 -0
  251. package/dist/services/usage-limit-policy.service.d.ts +12 -0
  252. package/dist/services/usage-limit-policy.service.js +22 -0
  253. package/dist/services/usage-limit-policy.service.test.d.ts +1 -0
  254. package/dist/services/usage-limit-policy.service.test.js +46 -0
  255. package/dist/services/usage.service.d.ts +27 -0
  256. package/dist/services/usage.service.js +77 -0
  257. package/dist/services/worksheet-content.service.d.ts +42 -0
  258. package/dist/services/worksheet-content.service.js +84 -0
  259. package/dist/services/worksheet-content.service.test.d.ts +1 -0
  260. package/dist/services/worksheet-content.service.test.js +69 -0
  261. package/dist/services/worksheet-generation.service.d.ts +91 -0
  262. package/dist/services/worksheet-generation.service.js +95 -0
  263. package/dist/services/worksheet-generation.service.test.d.ts +1 -0
  264. package/dist/services/worksheet-generation.service.test.js +20 -0
  265. package/dist/services/worksheet.service.d.ts +385 -0
  266. package/dist/services/worksheet.service.js +596 -0
  267. package/dist/services/workspace/workspace-analytics.service.d.ts +24 -0
  268. package/dist/services/workspace/workspace-analytics.service.js +95 -0
  269. package/dist/services/workspace/workspace-kb.service.d.ts +40 -0
  270. package/dist/services/workspace/workspace-kb.service.js +184 -0
  271. package/dist/services/workspace/workspace.service.d.ts +263 -0
  272. package/dist/services/workspace/workspace.service.js +401 -0
  273. package/dist/services/workspace-analytics.service.d.ts +24 -0
  274. package/dist/services/workspace-analytics.service.js +95 -0
  275. package/dist/services/workspace-kb.service.d.ts +40 -0
  276. package/dist/services/workspace-kb.service.js +184 -0
  277. package/dist/services/workspace-progress.service.d.ts +27 -0
  278. package/dist/services/workspace-progress.service.js +56 -0
  279. package/dist/services/workspace-progress.service.test.d.ts +1 -0
  280. package/dist/services/workspace-progress.service.test.js +49 -0
  281. package/dist/services/workspace.service.d.ts +307 -0
  282. package/dist/services/workspace.service.js +390 -0
  283. package/dist/trpc.d.ts +12 -4
  284. package/dist/trpc.js +7 -13
  285. package/package.json +5 -6
  286. package/prisma/migrations/20260509000001_add_knowledge_base/migration.sql +99 -0
  287. package/prisma/migrations/20260509000002_curate_knowledge_base/migration.sql +52 -0
  288. package/prisma/migrations/20260522000000_add_notes/migration.sql +27 -0
  289. package/prisma/migrations/20260524000000_remove_notes/migration.sql +3 -0
  290. package/prisma/schema.prisma +150 -48
  291. package/prisma/seed.mjs +67 -0
  292. package/scripts/debug/README.md +4 -0
  293. package/src/README.md +63 -0
  294. package/src/context.ts +33 -3
  295. package/src/lib/ai/config.ts +34 -0
  296. package/src/lib/ai/embedding-client.ts +47 -0
  297. package/src/lib/ai/index.ts +65 -0
  298. package/src/lib/ai/inference-backend/client.ts +479 -0
  299. package/src/lib/ai/inference-backend/mocks.ts +171 -0
  300. package/src/lib/ai/inference-backend/types.ts +50 -0
  301. package/src/lib/ai/json-parse.ts +35 -0
  302. package/src/lib/ai/llm-client.ts +54 -0
  303. package/src/lib/ai/mock.ts +12 -0
  304. package/src/lib/ai/types.ts +11 -0
  305. package/src/lib/chunking.ts +81 -0
  306. package/src/lib/curated-kb-seed.ts +164 -0
  307. package/src/lib/email.ts +77 -115
  308. package/src/lib/embeddings.ts +9 -0
  309. package/src/lib/ensure-curated-kb-catalog.ts +60 -0
  310. package/src/lib/env.ts +2 -7
  311. package/src/lib/inference.ts +1 -21
  312. package/src/lib/kb-meta.ts +81 -0
  313. package/src/lib/pdf.ts +23 -0
  314. package/src/lib/workspace-kb.ts +7 -0
  315. package/src/repositories/artifact.repository.ts +55 -0
  316. package/src/repositories/base.repository.ts +19 -0
  317. package/src/repositories/invitation.repository.ts +53 -0
  318. package/src/repositories/notification.repository.ts +53 -0
  319. package/src/repositories/user.repository.ts +44 -0
  320. package/src/repositories/workspace-member.repository.ts +38 -0
  321. package/src/repositories/workspace.repository.ts +89 -0
  322. package/src/routers/_app.ts +4 -0
  323. package/src/routers/admin.ts +124 -692
  324. package/src/routers/annotations.ts +25 -203
  325. package/src/routers/artifactVersions.ts +32 -0
  326. package/src/routers/auth.ts +82 -520
  327. package/src/routers/chat.ts +42 -245
  328. package/src/routers/copilot.ts +41 -666
  329. package/src/routers/flashcards.ts +108 -404
  330. package/src/routers/knowledgeBase.ts +216 -0
  331. package/src/routers/members.ts +60 -782
  332. package/src/routers/notifications.ts +15 -117
  333. package/src/routers/payment.ts +37 -446
  334. package/src/routers/podcast.ts +36 -898
  335. package/src/routers/studyguide.ts +5 -144
  336. package/src/routers/worksheets.ts +171 -735
  337. package/src/routers/workspace.ts +141 -1108
  338. package/src/scripts/purge-deleted-users.ts +2 -2
  339. package/src/server.ts +10 -3
  340. package/src/{lib/activity_human_description.test.ts → services/activity/activity-human-description.service.test.ts} +1 -1
  341. package/src/{lib/activity_log_service.test.ts → services/activity/activity-log.service.test.ts} +1 -1
  342. package/src/{lib/activity_log_service.ts → services/activity/activity-log.service.ts} +2 -2
  343. package/src/services/admin/admin.service.ts +612 -0
  344. package/src/services/ai/ai-session.service.ts +5 -0
  345. package/src/services/artifacts/annotation.service.ts +189 -0
  346. package/src/services/artifacts/artifact-version.service.ts +151 -0
  347. package/src/services/artifacts/chat.service.ts +197 -0
  348. package/src/services/artifacts/study-guide.service.ts +72 -0
  349. package/src/services/auth/auth.service.ts +473 -0
  350. package/src/services/base.service.ts +19 -0
  351. package/src/services/billing/payment.service.ts +433 -0
  352. package/src/{lib/subscription_service.ts → services/billing/subscription.service.ts} +5 -5
  353. package/src/services/billing/usage.service.ts +207 -0
  354. package/src/services/content/copilot.service.ts +587 -0
  355. package/src/services/{flashcard-progress.service.ts → content/flashcard-progress.service.ts} +18 -12
  356. package/src/services/content/flashcard.service.ts +417 -0
  357. package/src/services/content/media-analysis.service.ts +561 -0
  358. package/src/services/content/podcast.service.ts +777 -0
  359. package/src/services/content/worksheet-content.service.test.ts +83 -0
  360. package/src/services/content/worksheet-content.service.ts +117 -0
  361. package/src/{lib/worksheet-generation.test.ts → services/content/worksheet-generation.service.test.ts} +3 -3
  362. package/src/services/content/worksheet.service.ts +751 -0
  363. package/src/services/knowledge/knowledge-base.service.ts +705 -0
  364. package/src/services/members/invitation.service.ts +427 -0
  365. package/src/services/members/member.service.ts +241 -0
  366. package/src/{lib/notification-service.test.ts → services/notifications/notification.service.test.ts} +2 -2
  367. package/src/{lib/notification-service.ts → services/notifications/notification.service.ts} +102 -1
  368. package/src/services/workspace/workspace-analytics.service.ts +107 -0
  369. package/src/services/workspace/workspace-kb.service.ts +273 -0
  370. package/src/services/workspace/workspace.service.ts +488 -0
  371. package/src/trpc.ts +7 -15
  372. package/src/lib/ai-session.ts +0 -704
  373. package/src/lib/usage_service.ts +0 -74
  374. package/src/lib/workspace-access.ts +0 -13
  375. /package/{check-difficulty.cjs → scripts/debug/check-difficulty.cjs} +0 -0
  376. /package/{check-questions.cjs → scripts/debug/check-questions.cjs} +0 -0
  377. /package/{db-summary.cjs → scripts/debug/db-summary.cjs} +0 -0
  378. /package/{mcq-test.cjs → scripts/debug/mcq-test.cjs} +0 -0
  379. /package/{test-generate.js → scripts/debug/test-generate.js} +0 -0
  380. /package/{test-ratio.cjs → scripts/debug/test-ratio.cjs} +0 -0
  381. /package/{zod-test.cjs → scripts/debug/zod-test.cjs} +0 -0
  382. /package/src/{lib/activity_human_description.ts → services/activity/activity-human-description.service.ts} +0 -0
  383. /package/src/{lib/worksheet-generation.ts → services/content/worksheet-generation.service.ts} +0 -0
@@ -0,0 +1,488 @@
1
+ import type { PrismaClient } from '@prisma/client';
2
+ import { TRPCError } from '@trpc/server';
3
+ import { BaseService } from '../base.service.js';
4
+ import { supabaseClient } from '../../lib/storage.js';
5
+ import PusherService from '../../lib/pusher.js';
6
+ import { ai } from '../../lib/ai/index.js';
7
+ import { workspaceKbService } from './workspace-kb.service.js';
8
+ import {
9
+ getAccountSummary,
10
+ invalidateUserBillingCache,
11
+ } from '../billing/usage.service.js';
12
+ import { getUserStorageLimit } from '../billing/subscription.service.js';
13
+ import { notifyWorkspaceDeleted } from '../notifications/notification.service.js';
14
+
15
+ export class WorkspaceService extends BaseService {
16
+ constructor(db: PrismaClient) {
17
+ super(db);
18
+ }
19
+
20
+ async list(userId: string, parentId: string | null) {
21
+ const workspaces = await this.db.workspace.findMany({
22
+ where: { ownerId: userId, folderId: parentId },
23
+ orderBy: { updatedAt: 'desc' },
24
+ });
25
+ const folders = await this.db.folder.findMany({
26
+ where: { ownerId: userId, parentId },
27
+ });
28
+ return { workspaces, folders };
29
+ }
30
+
31
+ async getTree(userId: string) {
32
+ const [allFolders, allWorkspaces] = await Promise.all([
33
+ this.db.folder.findMany({
34
+ where: { ownerId: userId },
35
+ select: {
36
+ id: true,
37
+ name: true,
38
+ parentId: true,
39
+ color: true,
40
+ markerColor: true,
41
+ updatedAt: true,
42
+ },
43
+ orderBy: { updatedAt: 'desc' },
44
+ }),
45
+ this.db.workspace.findMany({
46
+ where: { ownerId: userId },
47
+ select: {
48
+ id: true,
49
+ title: true,
50
+ folderId: true,
51
+ icon: true,
52
+ color: true,
53
+ markerColor: true,
54
+ updatedAt: true,
55
+ uploads: {
56
+ select: {
57
+ id: true,
58
+ name: true,
59
+ mimeType: true,
60
+ createdAt: true,
61
+ },
62
+ },
63
+ },
64
+ orderBy: { updatedAt: 'desc' },
65
+ }),
66
+ ]);
67
+
68
+ return { folders: allFolders, workspaces: allWorkspaces };
69
+ }
70
+
71
+ async create(
72
+ userId: string,
73
+ input: {
74
+ name: string;
75
+ description?: string;
76
+ parentId?: string;
77
+ markerColor?: string | null;
78
+ },
79
+ ) {
80
+ const ws = await this.db.workspace.create({
81
+ data: {
82
+ title: input.name,
83
+ description: input.description,
84
+ ownerId: userId,
85
+ folderId: input.parentId ?? null,
86
+ ...(input.markerColor !== undefined ? { markerColor: input.markerColor } : {}),
87
+ },
88
+ });
89
+
90
+ void ai.backend.initSession(ws.id, userId).catch((err) => {
91
+ this.logger.error('Failed to init AI session on workspace creation:', err);
92
+ });
93
+
94
+ void workspaceKbService.ensureWorkspaceKb(ws.id, userId, input.name).catch((err) => {
95
+ this.logger.error('Failed to create workspace knowledge base:', err);
96
+ });
97
+
98
+ invalidateUserBillingCache(userId);
99
+ void PusherService.emitLibraryUpdate(userId);
100
+
101
+ return ws;
102
+ }
103
+
104
+ async createFolder(
105
+ userId: string,
106
+ input: { name: string; color?: string; parentId?: string },
107
+ ) {
108
+ const folder = await this.db.folder.create({
109
+ data: {
110
+ name: input.name,
111
+ ownerId: userId,
112
+ color: input.color ?? '#9D00FF',
113
+ parentId: input.parentId ?? null,
114
+ },
115
+ });
116
+ await PusherService.emitLibraryUpdate(userId);
117
+ return folder;
118
+ }
119
+
120
+ async updateFolder(
121
+ userId: string,
122
+ input: { id: string; name?: string; markerColor?: string | null },
123
+ ) {
124
+ const folder = await this.db.folder.update({
125
+ where: { id: input.id },
126
+ data: { name: input.name, markerColor: input.markerColor },
127
+ });
128
+ await PusherService.emitLibraryUpdate(userId);
129
+ return folder;
130
+ }
131
+
132
+ async deleteFolder(userId: string, id: string) {
133
+ const folder = await this.db.folder.delete({ where: { id } });
134
+ await PusherService.emitLibraryUpdate(userId);
135
+ return folder;
136
+ }
137
+
138
+ async get(userId: string, id: string) {
139
+ const ws = await this.db.workspace.findFirst({
140
+ where: { id, ownerId: userId },
141
+ select: {
142
+ id: true,
143
+ title: true,
144
+ description: true,
145
+ icon: true,
146
+ color: true,
147
+ markerColor: true,
148
+ ownerId: true,
149
+ folderId: true,
150
+ fileBeingAnalyzed: true,
151
+ analysisProgress: true,
152
+ needsAnalysis: true,
153
+ createdAt: true,
154
+ updatedAt: true,
155
+ folder: { select: { id: true, name: true, color: true } },
156
+ uploads: {
157
+ select: {
158
+ id: true,
159
+ name: true,
160
+ mimeType: true,
161
+ size: true,
162
+ createdAt: true,
163
+ objectKey: true,
164
+ },
165
+ orderBy: { createdAt: 'desc' },
166
+ },
167
+ },
168
+ });
169
+ if (!ws) throw new TRPCError({ code: 'NOT_FOUND' });
170
+ return ws;
171
+ }
172
+
173
+ async getAccountSummary(userId: string) {
174
+ return getAccountSummary(userId);
175
+ }
176
+
177
+ async getStats(userId: string) {
178
+ const summary = await getAccountSummary(userId);
179
+ return summary.stats;
180
+ }
181
+
182
+ async update(
183
+ userId: string,
184
+ input: {
185
+ id: string;
186
+ name?: string;
187
+ description?: string;
188
+ markerColor?: string | null;
189
+ icon?: string;
190
+ },
191
+ ) {
192
+ const existed = await this.db.workspace.findFirst({
193
+ where: { id: input.id, ownerId: userId },
194
+ });
195
+ if (!existed) throw new TRPCError({ code: 'NOT_FOUND' });
196
+ const updated = await this.db.workspace.update({
197
+ where: { id: input.id },
198
+ data: {
199
+ title: input.name ?? existed.title,
200
+ description: input.description,
201
+ markerColor: input.markerColor !== undefined ? input.markerColor : existed.markerColor,
202
+ icon: input.icon ?? existed.icon,
203
+ },
204
+ });
205
+
206
+ await PusherService.emitLibraryUpdate(userId);
207
+ return updated;
208
+ }
209
+
210
+ async delete(userId: string, id: string) {
211
+ const workspaceToDelete = await this.db.workspace.findFirst({
212
+ where: { id, ownerId: userId },
213
+ select: {
214
+ id: true,
215
+ title: true,
216
+ ownerId: true,
217
+ members: { select: { userId: true } },
218
+ },
219
+ });
220
+
221
+ if (!workspaceToDelete) throw new TRPCError({ code: 'NOT_FOUND' });
222
+
223
+ const actor = await this.db.user.findUnique({
224
+ where: { id: userId },
225
+ select: { name: true, email: true },
226
+ });
227
+ const actorName = actor?.name || actor?.email || 'A user';
228
+
229
+ await notifyWorkspaceDeleted(this.db, {
230
+ recipientUserIds: workspaceToDelete.members.map((m) => m.userId),
231
+ actorUserId: userId,
232
+ actorName,
233
+ workspaceId: workspaceToDelete.id,
234
+ workspaceTitle: workspaceToDelete.title,
235
+ });
236
+
237
+ const deleted = await this.db.workspace.deleteMany({
238
+ where: { id, ownerId: userId },
239
+ });
240
+ if (deleted.count === 0) throw new TRPCError({ code: 'NOT_FOUND' });
241
+
242
+ await PusherService.emitLibraryUpdate(userId);
243
+ return true;
244
+ }
245
+
246
+ async getFolderInformation(userId: string, id: string) {
247
+ const folder = await this.db.folder.findFirst({
248
+ where: { id, ownerId: userId },
249
+ });
250
+ if (!folder) throw new TRPCError({ code: 'NOT_FOUND' });
251
+
252
+ const parents: typeof folder[] = [];
253
+ let current = folder;
254
+ while (current.parentId) {
255
+ const parent = await this.db.folder.findFirst({
256
+ where: { id: current.parentId, ownerId: userId },
257
+ });
258
+ if (!parent) break;
259
+ parents.push(parent);
260
+ current = parent;
261
+ }
262
+
263
+ return { folder, parents };
264
+ }
265
+
266
+ async getSharedWith(userId: string, id: string) {
267
+ const user = await this.db.user.findFirst({ where: { id: userId } });
268
+ if (!user || !user.email) throw new TRPCError({ code: 'NOT_FOUND' });
269
+ const sharedWith = await this.db.workspace.findMany({
270
+ where: { members: { some: { userId } } },
271
+ });
272
+ const invitations = await this.db.workspaceInvitation.findMany({
273
+ where: { email: user.email, acceptedAt: null },
274
+ include: { workspace: true },
275
+ });
276
+ return { shared: sharedWith, invitations };
277
+ }
278
+
279
+ async uploadFiles(
280
+ userId: string,
281
+ input: {
282
+ id: string;
283
+ files: Array<{ filename: string; contentType: string; size: number }>;
284
+ },
285
+ ) {
286
+ const ws = await this.db.workspace.findFirst({
287
+ where: { id: input.id, ownerId: userId },
288
+ });
289
+ if (!ws) throw new TRPCError({ code: 'NOT_FOUND' });
290
+
291
+ const workspaces = await this.db.workspace.findMany({
292
+ where: {
293
+ OR: [{ ownerId: userId }, { sharedWith: { some: { id: userId } } }],
294
+ },
295
+ });
296
+ const spaceUsed = await this.db.fileAsset.aggregate({
297
+ where: { workspaceId: { in: workspaces.map((w: any) => w.id) }, userId },
298
+ _sum: { size: true },
299
+ });
300
+ const storageLimit = await getUserStorageLimit(userId);
301
+ const totalSize = input.files.reduce((acc, file) => acc + file.size, 0);
302
+ if ((spaceUsed._sum?.size ?? 0) + totalSize > storageLimit) {
303
+ this.logger.warn(
304
+ `Storage limit exceeded for user ${userId}. Used: ${spaceUsed._sum?.size}, Tried to upload: ${totalSize}, Limit: ${storageLimit}`,
305
+ );
306
+ throw new TRPCError({
307
+ code: 'FORBIDDEN',
308
+ message: `Storage limit exceeded. Maximum allowed storage is ${(storageLimit / (1024 * 1024 * 1024)).toFixed(1)}GB.`,
309
+ });
310
+ }
311
+
312
+ const results: Array<{ fileId: string; uploadUrl: string }> = [];
313
+
314
+ for (const file of input.files) {
315
+ const record = await this.db.fileAsset.create({
316
+ data: {
317
+ userId,
318
+ name: file.filename,
319
+ mimeType: file.contentType,
320
+ size: file.size,
321
+ workspaceId: input.id,
322
+ },
323
+ });
324
+
325
+ const objectKey = `${userId}/${record.id}-${file.filename}`;
326
+ const { data: signedUrlData, error: signedUrlError } = await supabaseClient.storage
327
+ .from('media')
328
+ .createSignedUploadUrl(objectKey);
329
+
330
+ if (signedUrlError) {
331
+ throw new TRPCError({
332
+ code: 'INTERNAL_SERVER_ERROR',
333
+ message: `Failed to upload file`,
334
+ });
335
+ }
336
+
337
+ await this.db.fileAsset.update({
338
+ where: { id: record.id },
339
+ data: {
340
+ bucket: 'media',
341
+ objectKey,
342
+ },
343
+ });
344
+
345
+ results.push({
346
+ fileId: record.id,
347
+ uploadUrl: signedUrlData.signedUrl,
348
+ });
349
+ }
350
+
351
+ return results;
352
+ }
353
+
354
+ async deleteFiles(userId: string, input: { fileId: string[]; id: string }) {
355
+ const files = await this.db.fileAsset.findMany({
356
+ where: {
357
+ id: { in: input.fileId },
358
+ workspaceId: input.id,
359
+ userId,
360
+ },
361
+ });
362
+ for (const file of files) {
363
+ if (file.bucket && file.objectKey) {
364
+ supabaseClient.storage
365
+ .from(file.bucket)
366
+ .remove([file.objectKey])
367
+ .catch((err: unknown) => {
368
+ this.logger.error(
369
+ `Error deleting file ${file.objectKey} from bucket ${file.bucket}:`,
370
+ err,
371
+ );
372
+ });
373
+ }
374
+ }
375
+
376
+ await this.db.fileAsset.deleteMany({
377
+ where: {
378
+ id: { in: input.fileId },
379
+ workspaceId: input.id,
380
+ userId,
381
+ },
382
+ });
383
+ return true;
384
+ }
385
+
386
+ async getFileUploadUrl(
387
+ userId: string,
388
+ input: {
389
+ workspaceId: string;
390
+ filename: string;
391
+ contentType: string;
392
+ size: number;
393
+ },
394
+ ) {
395
+ const workspaces = await this.db.workspace.findMany({
396
+ where: {
397
+ OR: [{ ownerId: userId }, { sharedWith: { some: { id: userId } } }],
398
+ },
399
+ });
400
+ const spaceUsed = await this.db.fileAsset.aggregate({
401
+ where: { workspaceId: { in: workspaces.map((w: any) => w.id) }, userId },
402
+ _sum: { size: true },
403
+ });
404
+ const storageLimit = await getUserStorageLimit(userId);
405
+ if ((spaceUsed._sum?.size ?? 0) + input.size > storageLimit) {
406
+ this.logger.warn(
407
+ `Storage limit exceeded for user ${userId}. Used: ${spaceUsed._sum?.size}, Tried to upload: ${input.size}, Limit: ${storageLimit}`,
408
+ );
409
+ throw new TRPCError({
410
+ code: 'FORBIDDEN',
411
+ message: `Storage limit exceeded. Maximum allowed storage is ${(storageLimit / (1024 * 1024 * 1024)).toFixed(1)}GB.`,
412
+ });
413
+ }
414
+
415
+ const objectKey = `workspace_${userId}/${input.workspaceId}-file_${input.filename}`;
416
+ const fileAsset = await this.db.fileAsset.create({
417
+ data: {
418
+ workspaceId: input.workspaceId,
419
+ name: input.filename,
420
+ mimeType: input.contentType,
421
+ size: input.size,
422
+ userId,
423
+ bucket: 'media',
424
+ objectKey,
425
+ },
426
+ });
427
+ const { data: signedUrlData, error: signedUrlError } = await supabaseClient.storage
428
+ .from('media')
429
+ .createSignedUploadUrl(objectKey, { upsert: true });
430
+ if (signedUrlError) {
431
+ this.logger.error('Signed upload URL error:', signedUrlError);
432
+ throw new TRPCError({
433
+ code: 'INTERNAL_SERVER_ERROR',
434
+ message: `Failed to create upload URL: ${signedUrlError.message}`,
435
+ });
436
+ }
437
+
438
+ await this.db.workspace.update({
439
+ where: { id: input.workspaceId },
440
+ data: { needsAnalysis: true },
441
+ });
442
+
443
+ return {
444
+ fileId: fileAsset.id,
445
+ uploadUrl: signedUrlData.signedUrl,
446
+ };
447
+ }
448
+
449
+ async search(userId: string, input: { query: string; color?: string; limit: number }) {
450
+ const { query, color } = input;
451
+
452
+ const workspaces = await this.db.workspace.findMany({
453
+ where: {
454
+ ownerId: userId,
455
+ markerColor: color || undefined,
456
+ ...(query
457
+ ? {
458
+ OR: [
459
+ { title: { contains: query, mode: 'insensitive' as const } },
460
+ { description: { contains: query, mode: 'insensitive' as const } },
461
+ ],
462
+ }
463
+ : {}),
464
+ },
465
+ orderBy: { updatedAt: 'desc' },
466
+ take: input.limit,
467
+ });
468
+
469
+ const folders = await this.db.folder.findMany({
470
+ where: {
471
+ ownerId: userId,
472
+ markerColor: color || undefined,
473
+ ...(query ? { name: { contains: query, mode: 'insensitive' as const } } : {}),
474
+ },
475
+ orderBy: { updatedAt: 'desc' },
476
+ take: input.limit,
477
+ });
478
+
479
+ const results = [
480
+ ...workspaces.map((w: any) => ({ ...w, type: 'workspace' as const })),
481
+ ...folders.map((f: any) => ({ ...f, type: 'folder' as const, title: f.name })),
482
+ ]
483
+ .sort((a, b) => b.updatedAt.getTime() - a.updatedAt.getTime())
484
+ .slice(0, input.limit);
485
+
486
+ return results;
487
+ }
488
+ }
package/src/trpc.ts CHANGED
@@ -4,13 +4,13 @@ import { ActivityLogStatus } from "@prisma/client";
4
4
  import type { Context } from "./context.js";
5
5
  import { logger } from "./lib/logger.js";
6
6
  import { toTRPCError } from "./lib/errors.js";
7
- import { getUserUsage, getUserPlanLimits } from "./lib/usage_service.js";
7
+ import { getUserUsage, getUserPlanLimits } from "./services/billing/usage.service.js";
8
8
  import {
9
9
  getClientIp,
10
10
  isActivityLogEnabled,
11
11
  scheduleRecordActivity,
12
12
  truncateUserAgent,
13
- } from "./lib/activity_log_service.js";
13
+ } from "./services/activity/activity-log.service.js";
14
14
 
15
15
  /** Avoid logging the log viewers themselves (noise when browsing activity). */
16
16
  const SKIP_ACTIVITY_TRPC_PATHS = new Set([
@@ -95,12 +95,8 @@ const errorHandler = middleware(async ({ next }) => {
95
95
  /**
96
96
  * Middleware that enforces email verification
97
97
  */
98
- const isVerified = middleware(async ({ ctx, next }) => {
99
- const user = await ctx.db.user.findUnique({
100
- where: { id: (ctx.session as any).user.id },
101
- select: { emailVerified: true },
102
- });
103
-
98
+ const isVerified = middleware(({ ctx, next }) => {
99
+ const user = (ctx.session as { user?: { emailVerified?: boolean } })?.user;
104
100
  if (!user?.emailVerified) {
105
101
  throw new TRPCError({
106
102
  code: "FORBIDDEN",
@@ -168,13 +164,9 @@ const checkUsageLimit = middleware(async ({ ctx, next, path }) => {
168
164
  /**
169
165
  * Middleware that enforces system admin role
170
166
  */
171
- const isAdmin = middleware(async ({ ctx, next }) => {
172
- const user = await ctx.db.user.findUnique({
173
- where: { id: (ctx.session as any).user.id },
174
- include: { role: true },
175
- });
176
-
177
- if (user?.role?.name !== 'System Admin') {
167
+ const isAdmin = middleware(({ ctx, next }) => {
168
+ const user = (ctx.session as { user?: { isSystemAdmin?: boolean } })?.user;
169
+ if (!user?.isSystemAdmin) {
178
170
  throw new TRPCError({
179
171
  code: "FORBIDDEN",
180
172
  message: "You do not have permission to access this resource",