@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
@@ -1,4 +1,6 @@
1
- import PusherService from './pusher.js';
1
+ import type { PrismaClient } from '@prisma/client';
2
+ import PusherService from '../../lib/pusher.js';
3
+ import { BaseService } from '../base.service.js';
2
4
 
3
5
  type NotificationPriority = 'LOW' | 'NORMAL' | 'HIGH';
4
6
 
@@ -675,3 +677,102 @@ export async function notifyArtifactFailed(
675
677
  priority: 'HIGH',
676
678
  });
677
679
  }
680
+
681
+ /**
682
+ * Inbox-side operations (list, mark read, delete) used by the notifications
683
+ * router. Creation flows live in the standalone exported functions above and
684
+ * are called directly by other services.
685
+ */
686
+ export class NotificationService extends BaseService {
687
+ constructor(db: PrismaClient) {
688
+ super(db);
689
+ }
690
+
691
+ async list(
692
+ userId: string,
693
+ input: {
694
+ cursor?: string;
695
+ limit: number;
696
+ unreadOnly?: boolean;
697
+ types?: string[];
698
+ },
699
+ ) {
700
+ const { cursor, limit, unreadOnly, types } = input;
701
+
702
+ const items = await this.db.notification.findMany({
703
+ where: {
704
+ userId,
705
+ ...(unreadOnly ? { read: false } : {}),
706
+ ...(types?.length ? { type: { in: types } } : {}),
707
+ },
708
+ include: {
709
+ actor: { select: { id: true, name: true, email: true } },
710
+ workspace: { select: { id: true, title: true } },
711
+ },
712
+ orderBy: [{ createdAt: 'desc' }, { id: 'desc' }],
713
+ take: limit + 1,
714
+ ...(cursor ? { cursor: { id: cursor }, skip: 1 } : {}),
715
+ });
716
+
717
+ let nextCursor: string | undefined;
718
+ if (items.length > limit) {
719
+ const next = items.pop();
720
+ nextCursor = next?.id;
721
+ }
722
+
723
+ return { items, nextCursor };
724
+ }
725
+
726
+ async unreadCount(userId: string) {
727
+ const count = await this.db.notification.count({
728
+ where: { userId, read: false },
729
+ });
730
+ return { count };
731
+ }
732
+
733
+ async markRead(userId: string, id: string) {
734
+ const now = new Date();
735
+ await this.db.notification.updateMany({
736
+ where: { id, userId },
737
+ data: { read: true, readAt: now },
738
+ });
739
+
740
+ const unreadCount = await this.db.notification.count({
741
+ where: { userId, read: false },
742
+ });
743
+ await PusherService.emitNotificationReadState(userId, { unreadCount });
744
+
745
+ return { success: true };
746
+ }
747
+
748
+ async markManyRead(userId: string, ids: string[]) {
749
+ const now = new Date();
750
+ await this.db.notification.updateMany({
751
+ where: { id: { in: ids }, userId },
752
+ data: { read: true, readAt: now },
753
+ });
754
+
755
+ const unreadCount = await this.db.notification.count({
756
+ where: { userId, read: false },
757
+ });
758
+ await PusherService.emitNotificationReadState(userId, { unreadCount });
759
+
760
+ return { success: true };
761
+ }
762
+
763
+ async markAllRead(userId: string) {
764
+ const now = new Date();
765
+ await this.db.notification.updateMany({
766
+ where: { userId, read: false },
767
+ data: { read: true, readAt: now },
768
+ });
769
+
770
+ await PusherService.emitNotificationReadState(userId, { unreadCount: 0 });
771
+ return { success: true };
772
+ }
773
+
774
+ async delete(userId: string, id: string) {
775
+ await this.db.notification.deleteMany({ where: { id, userId } });
776
+ return { success: true };
777
+ }
778
+ }
@@ -0,0 +1,107 @@
1
+ import type { PrismaClient } from '@prisma/client';
2
+ import { BaseService } from '../base.service.js';
3
+
4
+ /**
5
+ * Study analytics: streaks, weekly activity, flashcard mastery, worksheet
6
+ * accuracy. Behavior-preserving extraction from `workspace.getStudyAnalytics`.
7
+ */
8
+ export class WorkspaceAnalyticsService extends BaseService {
9
+ constructor(db: PrismaClient) {
10
+ super(db);
11
+ }
12
+
13
+ async getStudyAnalytics(userId: string) {
14
+ const flashcardProgress = await this.db.flashcardProgress.findMany({
15
+ where: { userId },
16
+ select: { lastStudiedAt: true },
17
+ });
18
+
19
+ const worksheetProgress = await this.db.worksheetQuestionProgress.findMany({
20
+ where: { userId },
21
+ select: { updatedAt: true, completedAt: true },
22
+ });
23
+
24
+ const studyDays = new Set<string>();
25
+ for (const fp of flashcardProgress) {
26
+ if (fp.lastStudiedAt) {
27
+ studyDays.add(fp.lastStudiedAt.toISOString().split('T')[0]);
28
+ }
29
+ }
30
+ for (const wp of worksheetProgress) {
31
+ if (wp.completedAt) {
32
+ studyDays.add(wp.completedAt.toISOString().split('T')[0]);
33
+ } else {
34
+ studyDays.add(wp.updatedAt.toISOString().split('T')[0]);
35
+ }
36
+ }
37
+
38
+ const sortedDays = [...studyDays].sort().reverse();
39
+ let streak = 0;
40
+
41
+ if (sortedDays.length > 0) {
42
+ const today = new Date();
43
+ today.setHours(0, 0, 0, 0);
44
+ const yesterday = new Date(today);
45
+ yesterday.setDate(yesterday.getDate() - 1);
46
+
47
+ const todayStr = today.toISOString().split('T')[0];
48
+ const yesterdayStr = yesterday.toISOString().split('T')[0];
49
+
50
+ if (sortedDays[0] === todayStr || sortedDays[0] === yesterdayStr) {
51
+ streak = 1;
52
+ for (let i = 1; i < sortedDays.length; i++) {
53
+ const current = new Date(sortedDays[i - 1]);
54
+ const prev = new Date(sortedDays[i]);
55
+ const diffDays = (current.getTime() - prev.getTime()) / (1000 * 60 * 60 * 24);
56
+ if (diffDays === 1) {
57
+ streak++;
58
+ } else {
59
+ break;
60
+ }
61
+ }
62
+ }
63
+ }
64
+
65
+ const weeklyActivity: boolean[] = [];
66
+ const today = new Date();
67
+ today.setHours(0, 0, 0, 0);
68
+ for (let i = 6; i >= 0; i--) {
69
+ const d = new Date(today);
70
+ d.setDate(d.getDate() - i);
71
+ const dayStr = d.toISOString().split('T')[0];
72
+ weeklyActivity.push(studyDays.has(dayStr));
73
+ }
74
+
75
+ const totalCards = await this.db.flashcardProgress.count({ where: { userId } });
76
+ const masteredCards = await this.db.flashcardProgress.count({
77
+ where: { userId, masteryLevel: { gte: 80 } },
78
+ });
79
+ const dueCards = await this.db.flashcardProgress.count({
80
+ where: { userId, nextReviewAt: { lte: new Date() } },
81
+ });
82
+
83
+ const completedQuestions = await this.db.worksheetQuestionProgress.count({
84
+ where: { userId, completedAt: { not: null } },
85
+ });
86
+ const correctQuestions = await this.db.worksheetQuestionProgress.count({
87
+ where: { userId, correct: true },
88
+ });
89
+
90
+ return {
91
+ streak,
92
+ totalStudyDays: studyDays.size,
93
+ weeklyActivity,
94
+ flashcards: {
95
+ total: totalCards,
96
+ mastered: masteredCards,
97
+ dueForReview: dueCards,
98
+ },
99
+ worksheets: {
100
+ completed: completedQuestions,
101
+ correct: correctQuestions,
102
+ accuracy:
103
+ completedQuestions > 0 ? Math.round((correctQuestions / completedQuestions) * 100) : 0,
104
+ },
105
+ };
106
+ }
107
+ }
@@ -0,0 +1,273 @@
1
+ import type { PrismaClient } from '@prisma/client';
2
+ import { BaseService } from '../base.service.js';
3
+ import { prisma } from '../../lib/prisma.js';
4
+ import {
5
+ DEFAULT_EMBEDDING_MODEL,
6
+ embedQuery,
7
+ embedTexts,
8
+ toVectorLiteral,
9
+ } from '../../lib/embeddings.js';
10
+ import { chunkText } from '../../lib/chunking.js';
11
+ import { WORKSPACE_KB_KIND } from '../../lib/workspace-kb.js';
12
+
13
+ export interface WorkspaceKbMatch {
14
+ chunkId: string;
15
+ documentId: string;
16
+ documentName: string;
17
+ chunkIndex: number;
18
+ content: string;
19
+ score: number;
20
+ }
21
+
22
+ export class WorkspaceKbService extends BaseService {
23
+ constructor(db: PrismaClient) {
24
+ super(db);
25
+ }
26
+
27
+ /** Returns the single auto-managed KB for a workspace, creating it if needed. */
28
+ async ensureWorkspaceKb(workspaceId: string, userId: string, title?: string) {
29
+ const existingLink = await this.db.workspaceKnowledgeBase.findFirst({
30
+ where: {
31
+ workspaceId,
32
+ knowledgeBase: {
33
+ isCurated: false,
34
+ meta: {
35
+ path: ['kind'],
36
+ equals: WORKSPACE_KB_KIND,
37
+ },
38
+ },
39
+ },
40
+ include: { knowledgeBase: true },
41
+ });
42
+
43
+ if (existingLink) {
44
+ return existingLink.knowledgeBase;
45
+ }
46
+
47
+ const workspace = await this.db.workspace.findUnique({
48
+ where: { id: workspaceId },
49
+ select: { title: true },
50
+ });
51
+
52
+ const kb = await this.db.knowledgeBase.create({
53
+ data: {
54
+ createdById: userId,
55
+ name: `${title ?? workspace?.title ?? 'Workspace'} Knowledge Base`,
56
+ description: 'Private knowledge base for this workspace',
57
+ isCurated: false,
58
+ embeddingModel: DEFAULT_EMBEDDING_MODEL,
59
+ embeddingDim: 1536,
60
+ meta: { kind: WORKSPACE_KB_KIND, workspaceId },
61
+ },
62
+ });
63
+
64
+ await this.db.workspaceKnowledgeBase.create({
65
+ data: {
66
+ workspaceId,
67
+ knowledgeBaseId: kb.id,
68
+ addedById: userId,
69
+ },
70
+ });
71
+
72
+ return kb;
73
+ }
74
+
75
+ async getWorkspaceKbId(workspaceId: string): Promise<string | null> {
76
+ const link = await this.db.workspaceKnowledgeBase.findFirst({
77
+ where: {
78
+ workspaceId,
79
+ knowledgeBase: {
80
+ isCurated: false,
81
+ meta: { path: ['kind'], equals: WORKSPACE_KB_KIND },
82
+ },
83
+ },
84
+ select: { knowledgeBaseId: true },
85
+ });
86
+ return link?.knowledgeBaseId ?? null;
87
+ }
88
+
89
+ async indexText(
90
+ workspaceId: string,
91
+ userId: string,
92
+ input: {
93
+ sourceType: string;
94
+ sourceId: string;
95
+ name: string;
96
+ text: string;
97
+ },
98
+ ) {
99
+ const trimmed = input.text.trim();
100
+ if (!trimmed) return;
101
+
102
+ try {
103
+ const kb = await this.ensureWorkspaceKb(workspaceId, userId);
104
+ const docs = await this.db.knowledgeBaseDocument.findMany({
105
+ where: { knowledgeBaseId: kb.id },
106
+ select: { id: true, meta: true },
107
+ });
108
+
109
+ const existing = docs.find((doc) => {
110
+ const meta = doc.meta as { sourceType?: string; sourceId?: string } | null;
111
+ return meta?.sourceType === input.sourceType && meta?.sourceId === input.sourceId;
112
+ });
113
+
114
+ let documentId: string;
115
+ if (existing) {
116
+ documentId = existing.id;
117
+ await this.db.knowledgeBaseChunk.deleteMany({ where: { documentId } });
118
+ await this.db.knowledgeBaseDocument.update({
119
+ where: { id: documentId },
120
+ data: {
121
+ name: input.name,
122
+ status: 'PROCESSING',
123
+ errorMessage: null,
124
+ },
125
+ });
126
+ } else {
127
+ const created = await this.db.knowledgeBaseDocument.create({
128
+ data: {
129
+ knowledgeBaseId: kb.id,
130
+ name: input.name,
131
+ mimeType: 'text/plain',
132
+ size: Buffer.byteLength(trimmed, 'utf8'),
133
+ status: 'PROCESSING',
134
+ meta: {
135
+ sourceType: input.sourceType,
136
+ sourceId: input.sourceId,
137
+ },
138
+ },
139
+ });
140
+ documentId = created.id;
141
+ }
142
+
143
+ await this.db.knowledgeBase.update({
144
+ where: { id: kb.id },
145
+ data: { status: 'INDEXING' },
146
+ });
147
+
148
+ const chunks = chunkText(trimmed, { targetTokens: 500, overlapTokens: 60 });
149
+ if (chunks.length === 0) return;
150
+
151
+ const embeddings = await embedTexts(
152
+ chunks.map((c) => c.content),
153
+ { model: kb.embeddingModel },
154
+ );
155
+
156
+ for (let i = 0; i < chunks.length; i += 1) {
157
+ const chunk = chunks[i];
158
+ const vec = toVectorLiteral(embeddings[i] ?? []);
159
+ await this.db.$executeRawUnsafe(
160
+ `INSERT INTO "KnowledgeBaseChunk"
161
+ ("id", "documentId", "knowledgeBaseId", "chunkIndex",
162
+ "pageNumber", "content", "tokenCount", "embedding", "createdAt")
163
+ VALUES (
164
+ gen_random_uuid()::text,
165
+ $1, $2, $3, $4, $5, $6, $7::vector, NOW()
166
+ )`,
167
+ documentId,
168
+ kb.id,
169
+ chunk.index,
170
+ null,
171
+ chunk.content,
172
+ chunk.approxTokens,
173
+ vec,
174
+ );
175
+ }
176
+
177
+ await this.db.knowledgeBaseDocument.update({
178
+ where: { id: documentId },
179
+ data: { status: 'READY', numChunks: chunks.length },
180
+ });
181
+ await this.db.knowledgeBase.update({
182
+ where: { id: kb.id },
183
+ data: { status: 'READY' },
184
+ });
185
+ } catch (err) {
186
+ this.logger.error(
187
+ `Failed to index ${input.sourceType}:${input.sourceId} into workspace KB`,
188
+ err,
189
+ );
190
+ }
191
+ }
192
+
193
+ /** Fire-and-forget indexing — never blocks the caller. */
194
+ indexTextAsync(
195
+ workspaceId: string,
196
+ userId: string,
197
+ input: Parameters<WorkspaceKbService['indexText']>[2],
198
+ ) {
199
+ void this.indexText(workspaceId, userId, input);
200
+ }
201
+
202
+ async search(
203
+ workspaceId: string,
204
+ query: string,
205
+ topK = 5,
206
+ ): Promise<WorkspaceKbMatch[]> {
207
+ const kbId = await this.getWorkspaceKbId(workspaceId);
208
+
209
+ if (!kbId) return [];
210
+
211
+ const kb = await this.db.knowledgeBase.findUnique({ where: { id: kbId } });
212
+ if (!kb) return [];
213
+
214
+
215
+ const k = Math.min(Math.max(topK, 1), 20);
216
+ const queryVec = await embedQuery(query, { model: kb.embeddingModel });
217
+ const vecLiteral = toVectorLiteral(queryVec);
218
+
219
+ const rows = await this.db.$queryRawUnsafe<
220
+ Array<{
221
+ id: string;
222
+ documentId: string;
223
+ chunkIndex: number;
224
+ content: string;
225
+ distance: number;
226
+ documentName: string;
227
+ }>
228
+ >(
229
+ `SELECT c."id", c."documentId", c."chunkIndex", c."content",
230
+ (c."embedding" <=> $1::vector) AS "distance",
231
+ d."name" AS "documentName"
232
+ FROM "KnowledgeBaseChunk" c
233
+ JOIN "KnowledgeBaseDocument" d ON d."id" = c."documentId"
234
+ WHERE c."knowledgeBaseId" = $2
235
+ AND c."embedding" IS NOT NULL
236
+ ORDER BY c."embedding" <=> $1::vector
237
+ LIMIT ${k}`,
238
+ vecLiteral,
239
+ kbId,
240
+ );
241
+
242
+
243
+ return rows.map((r) => ({
244
+ chunkId: r.id,
245
+ documentId: r.documentId,
246
+ documentName: r.documentName,
247
+ chunkIndex: r.chunkIndex,
248
+ content: r.content,
249
+ score: 1 - Number(r.distance),
250
+ }));
251
+ }
252
+
253
+ formatContext(matches: WorkspaceKbMatch[]): string {
254
+ if (matches.length === 0) return '';
255
+ return matches
256
+ .map(
257
+ (m, i) =>
258
+ `[${i + 1}] (${m.documentName}, chunk ${m.chunkIndex})\n${m.content}`,
259
+ )
260
+ .join('\n\n---\n\n');
261
+ }
262
+
263
+ async retrieveContext(
264
+ workspaceId: string,
265
+ query: string,
266
+ topK = 5,
267
+ ): Promise<string> {
268
+ const matches = await this.search(workspaceId, query, topK);
269
+ return this.formatContext(matches);
270
+ }
271
+ }
272
+
273
+ export const workspaceKbService = new WorkspaceKbService(prisma);