@goscribe/server 1.3.3 → 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
@@ -0,0 +1,596 @@
1
+ import { TRPCError } from '@trpc/server';
2
+ import { BaseService } from './base.service.js';
3
+ import { ArtifactType } from '../lib/constants.js';
4
+ import { workspaceAccessWhere } from '../repositories/workspace.repository.js';
5
+ import PusherService from '../lib/pusher.js';
6
+ import { aiSessionService } from './ai-session.service.js';
7
+ import { notifyArtifactFailed, notifyArtifactReady } from './notification.service.js';
8
+ import { mergeWorksheetGenerationConfig, normalizeWorksheetProblemForDb, parsePresetConfig, } from './worksheet-generation.service.js';
9
+ import { mergeWorksheetQuestionProgress } from './worksheet-content.service.js';
10
+ const Difficulty = { EASY: 'EASY', MEDIUM: 'MEDIUM', HARD: 'HARD' };
11
+ const QuestionType = {
12
+ MULTIPLE_CHOICE: 'MULTIPLE_CHOICE',
13
+ TEXT: 'TEXT',
14
+ NUMERIC: 'NUMERIC',
15
+ TRUE_FALSE: 'TRUE_FALSE',
16
+ MATCHING: 'MATCHING',
17
+ FILL_IN_THE_BLANK: 'FILL_IN_THE_BLANK',
18
+ };
19
+ export class WorksheetService extends BaseService {
20
+ constructor(db) {
21
+ super(db);
22
+ }
23
+ async list(userId, workspaceId) {
24
+ const worksheets = await this.db.artifact.findMany({
25
+ where: { workspaceId, type: ArtifactType.WORKSHEET },
26
+ include: {
27
+ versions: { orderBy: { version: 'desc' }, take: 1 },
28
+ questions: true,
29
+ },
30
+ orderBy: { updatedAt: 'desc' },
31
+ });
32
+ if (!worksheets)
33
+ throw new TRPCError({ code: 'NOT_FOUND' });
34
+ const allQuestionIds = worksheets.flatMap((w) => w.questions.map((q) => q.id));
35
+ if (allQuestionIds.length === 0)
36
+ return worksheets;
37
+ const progress = await this.db.worksheetQuestionProgress.findMany({
38
+ where: { userId, worksheetQuestionId: { in: allQuestionIds } },
39
+ });
40
+ return worksheets.map((w) => ({
41
+ ...w,
42
+ questions: mergeWorksheetQuestionProgress(w.questions, progress),
43
+ }));
44
+ }
45
+ async listPresets(userId, workspaceId) {
46
+ const ws = await this.db.workspace.findFirst({
47
+ where: { id: workspaceId, ...workspaceAccessWhere(userId) },
48
+ });
49
+ if (!ws)
50
+ throw new TRPCError({ code: 'NOT_FOUND' });
51
+ return this.db.worksheetPreset.findMany({
52
+ where: {
53
+ OR: [
54
+ { isSystem: true },
55
+ {
56
+ userId,
57
+ OR: [{ workspaceId }, { workspaceId: null }],
58
+ },
59
+ ],
60
+ },
61
+ orderBy: [{ isSystem: 'desc' }, { name: 'asc' }],
62
+ });
63
+ }
64
+ async createPreset(userId, input) {
65
+ if (input.workspaceId) {
66
+ const ws = await this.db.workspace.findFirst({
67
+ where: { id: input.workspaceId, ...workspaceAccessWhere(userId) },
68
+ });
69
+ if (!ws)
70
+ throw new TRPCError({ code: 'NOT_FOUND' });
71
+ }
72
+ const config = parsePresetConfig(input.config);
73
+ return this.db.worksheetPreset.create({
74
+ data: {
75
+ userId,
76
+ workspaceId: input.workspaceId ?? null,
77
+ name: input.name,
78
+ isSystem: false,
79
+ config: config,
80
+ },
81
+ });
82
+ }
83
+ async updatePreset(userId, input) {
84
+ const existing = await this.db.worksheetPreset.findFirst({
85
+ where: { id: input.id, userId, isSystem: false },
86
+ });
87
+ if (!existing) {
88
+ throw new TRPCError({ code: 'NOT_FOUND', message: 'Preset not found or read-only' });
89
+ }
90
+ const data = {};
91
+ if (input.name !== undefined)
92
+ data.name = input.name;
93
+ if (input.config !== undefined)
94
+ data.config = parsePresetConfig(input.config);
95
+ return this.db.worksheetPreset.update({
96
+ where: { id: input.id },
97
+ data,
98
+ });
99
+ }
100
+ async deletePreset(userId, id) {
101
+ const existing = await this.db.worksheetPreset.findFirst({
102
+ where: { id, userId, isSystem: false },
103
+ });
104
+ if (!existing) {
105
+ throw new TRPCError({ code: 'NOT_FOUND', message: 'Preset not found or read-only' });
106
+ }
107
+ await this.db.worksheetPreset.delete({ where: { id } });
108
+ return true;
109
+ }
110
+ async create(userId, input) {
111
+ const workspace = await this.db.workspace.findFirst({
112
+ where: { id: input.workspaceId, ownerId: userId },
113
+ });
114
+ if (!workspace)
115
+ throw new TRPCError({ code: 'NOT_FOUND' });
116
+ const { problems } = input;
117
+ return this.db.artifact.create({
118
+ data: {
119
+ workspaceId: input.workspaceId,
120
+ type: ArtifactType.WORKSHEET,
121
+ title: input.title,
122
+ difficulty: input.difficulty,
123
+ estimatedTime: input.estimatedTime,
124
+ createdById: userId,
125
+ questions: problems
126
+ ? {
127
+ create: problems.map((problem, index) => ({
128
+ prompt: problem.question,
129
+ answer: problem.answer,
130
+ type: problem.type,
131
+ order: index,
132
+ meta: problem.options ? { options: problem.options } : undefined,
133
+ })),
134
+ }
135
+ : undefined,
136
+ },
137
+ include: { questions: true },
138
+ });
139
+ }
140
+ async get(userId, worksheetId) {
141
+ const worksheet = await this.db.artifact.findFirst({
142
+ where: {
143
+ id: worksheetId,
144
+ type: ArtifactType.WORKSHEET,
145
+ workspace: workspaceAccessWhere(userId),
146
+ },
147
+ include: { questions: true },
148
+ orderBy: { updatedAt: 'desc' },
149
+ });
150
+ if (!worksheet)
151
+ throw new TRPCError({ code: 'NOT_FOUND' });
152
+ const questionIds = worksheet.questions.map((q) => q.id);
153
+ if (questionIds.length === 0)
154
+ return worksheet;
155
+ const progress = await this.db.worksheetQuestionProgress.findMany({
156
+ where: { userId, worksheetQuestionId: { in: questionIds } },
157
+ });
158
+ return {
159
+ ...worksheet,
160
+ questions: mergeWorksheetQuestionProgress(worksheet.questions, progress),
161
+ };
162
+ }
163
+ async createWorksheetQuestion(userId, input) {
164
+ const worksheet = await this.db.artifact.findFirst({
165
+ where: {
166
+ id: input.worksheetId,
167
+ type: ArtifactType.WORKSHEET,
168
+ workspace: workspaceAccessWhere(userId),
169
+ },
170
+ });
171
+ if (!worksheet)
172
+ throw new TRPCError({ code: 'NOT_FOUND' });
173
+ return this.db.worksheetQuestion.create({
174
+ data: {
175
+ artifactId: input.worksheetId,
176
+ prompt: input.prompt,
177
+ answer: input.answer,
178
+ type: (input.type ?? QuestionType.TEXT),
179
+ difficulty: (input.difficulty ?? Difficulty.MEDIUM),
180
+ order: input.order ?? 0,
181
+ meta: input.meta,
182
+ },
183
+ });
184
+ }
185
+ async updateWorksheetQuestion(userId, input) {
186
+ const q = await this.db.worksheetQuestion.findFirst({
187
+ where: {
188
+ id: input.worksheetQuestionId,
189
+ artifact: {
190
+ type: ArtifactType.WORKSHEET,
191
+ workspace: workspaceAccessWhere(userId),
192
+ },
193
+ },
194
+ });
195
+ if (!q)
196
+ throw new TRPCError({ code: 'NOT_FOUND' });
197
+ return this.db.worksheetQuestion.update({
198
+ where: { id: input.worksheetQuestionId },
199
+ data: {
200
+ prompt: input.prompt ?? q.prompt,
201
+ answer: input.answer ?? q.answer,
202
+ type: (input.type ?? q.type),
203
+ difficulty: (input.difficulty ?? q.difficulty),
204
+ order: input.order ?? q.order,
205
+ meta: (input.meta ?? q.meta),
206
+ },
207
+ });
208
+ }
209
+ async deleteWorksheetQuestion(userId, worksheetQuestionId) {
210
+ const q = await this.db.worksheetQuestion.findFirst({
211
+ where: {
212
+ id: worksheetQuestionId,
213
+ artifact: { workspace: workspaceAccessWhere(userId) },
214
+ },
215
+ });
216
+ if (!q)
217
+ throw new TRPCError({ code: 'NOT_FOUND' });
218
+ await this.db.worksheetQuestion.delete({ where: { id: worksheetQuestionId } });
219
+ return true;
220
+ }
221
+ async updateProblemStatus(userId, input) {
222
+ const question = await this.db.worksheetQuestion.findFirst({
223
+ where: {
224
+ id: input.problemId,
225
+ artifact: {
226
+ type: ArtifactType.WORKSHEET,
227
+ workspace: workspaceAccessWhere(userId),
228
+ },
229
+ },
230
+ });
231
+ if (!question)
232
+ throw new TRPCError({ code: 'NOT_FOUND' });
233
+ return this.db.worksheetQuestionProgress.upsert({
234
+ where: {
235
+ worksheetQuestionId_userId: {
236
+ worksheetQuestionId: input.problemId,
237
+ userId,
238
+ },
239
+ },
240
+ create: {
241
+ worksheetQuestionId: input.problemId,
242
+ userId,
243
+ modified: input.completed,
244
+ userAnswer: input.answer,
245
+ correct: input.correct,
246
+ completedAt: input.completed ? new Date() : null,
247
+ attempts: 1,
248
+ },
249
+ update: {
250
+ modified: input.completed,
251
+ userAnswer: input.answer,
252
+ completedAt: input.completed ? new Date() : null,
253
+ attempts: { increment: 1 },
254
+ correct: input.correct,
255
+ },
256
+ });
257
+ }
258
+ async getProgress(userId, worksheetId) {
259
+ const worksheet = await this.db.artifact.findFirst({
260
+ where: {
261
+ id: worksheetId,
262
+ type: ArtifactType.WORKSHEET,
263
+ workspace: workspaceAccessWhere(userId),
264
+ },
265
+ });
266
+ if (!worksheet)
267
+ throw new TRPCError({ code: 'NOT_FOUND' });
268
+ const questions = await this.db.worksheetQuestion.findMany({
269
+ where: { artifactId: worksheetId },
270
+ select: { id: true },
271
+ });
272
+ const questionIds = questions.map((q) => q.id);
273
+ return this.db.worksheetQuestionProgress.findMany({
274
+ where: { userId, worksheetQuestionId: { in: questionIds } },
275
+ });
276
+ }
277
+ async update(userId, input) {
278
+ const { id, problems, ...updateData } = input;
279
+ const existingWorksheet = await this.db.artifact.findFirst({
280
+ where: {
281
+ id,
282
+ type: ArtifactType.WORKSHEET,
283
+ workspace: workspaceAccessWhere(userId),
284
+ },
285
+ });
286
+ if (!existingWorksheet)
287
+ throw new TRPCError({ code: 'NOT_FOUND' });
288
+ if (problems) {
289
+ await this.db.worksheetQuestion.deleteMany({ where: { artifactId: id } });
290
+ await this.db.worksheetQuestion.createMany({
291
+ data: problems.map((problem, index) => ({
292
+ artifactId: id,
293
+ prompt: problem.question,
294
+ answer: problem.answer,
295
+ type: problem.type,
296
+ order: index,
297
+ meta: problem.options ? { options: problem.options } : undefined,
298
+ })),
299
+ });
300
+ }
301
+ const processedUpdateData = {
302
+ ...updateData,
303
+ difficulty: updateData.difficulty,
304
+ };
305
+ return this.db.artifact.update({
306
+ where: { id },
307
+ data: processedUpdateData,
308
+ include: {
309
+ questions: { orderBy: { order: 'asc' } },
310
+ },
311
+ });
312
+ }
313
+ async delete(userId, id) {
314
+ const deleted = await this.db.artifact.deleteMany({
315
+ where: {
316
+ id,
317
+ type: ArtifactType.WORKSHEET,
318
+ workspace: workspaceAccessWhere(userId),
319
+ },
320
+ });
321
+ if (deleted.count === 0)
322
+ throw new TRPCError({ code: 'NOT_FOUND' });
323
+ return true;
324
+ }
325
+ async generateFromPrompt(userId, input) {
326
+ const workspace = await this.db.workspace.findFirst({
327
+ where: { id: input.workspaceId, ...workspaceAccessWhere(userId) },
328
+ });
329
+ if (!workspace)
330
+ throw new TRPCError({ code: 'NOT_FOUND' });
331
+ let presetRow = null;
332
+ if (input.presetId) {
333
+ presetRow = await this.db.worksheetPreset.findFirst({
334
+ where: {
335
+ id: input.presetId,
336
+ OR: [
337
+ { isSystem: true },
338
+ {
339
+ userId,
340
+ OR: [{ workspaceId: input.workspaceId }, { workspaceId: null }],
341
+ },
342
+ ],
343
+ },
344
+ });
345
+ if (!presetRow)
346
+ throw new TRPCError({ code: 'NOT_FOUND', message: 'Preset not found' });
347
+ }
348
+ const presetConfig = presetRow?.config ? parsePresetConfig(presetRow.config) : undefined;
349
+ const resolved = mergeWorksheetGenerationConfig(presetConfig, input.configOverride ?? undefined, {
350
+ numQuestions: input.numQuestions,
351
+ difficulty: input.difficulty,
352
+ mode: input.mode,
353
+ });
354
+ const difficultyUpper = resolved.difficulty.toUpperCase();
355
+ console.log('[DEBUG TRPC PAYLOAD] input =', input);
356
+ console.log('[DEBUG TRPC PAYLOAD] presetConfig =', presetConfig);
357
+ console.log('[DEBUG TRPC PAYLOAD] legacy merged legacy values:', {
358
+ numQuestions: input.numQuestions,
359
+ difficulty: input.difficulty,
360
+ mode: input.mode,
361
+ });
362
+ console.log('[DEBUG TRPC PAYLOAD] RESOLVED =', resolved);
363
+ const worksheetConfigSnapshot = {
364
+ mode: resolved.mode,
365
+ presetId: input.presetId ?? null,
366
+ presetName: presetRow?.name ?? null,
367
+ numQuestions: resolved.numQuestions,
368
+ difficulty: resolved.difficulty,
369
+ mcqRatio: resolved.mcqRatio ?? null,
370
+ questionTypes: resolved.questionTypes ?? null,
371
+ };
372
+ await PusherService.emitWorksheetGenerationStart(input.workspaceId);
373
+ const artifact = await this.db.artifact.create({
374
+ data: {
375
+ workspaceId: input.workspaceId,
376
+ type: ArtifactType.WORKSHEET,
377
+ title: input.title ||
378
+ (resolved.mode === 'quiz'
379
+ ? `Quiz - ${new Date().toLocaleString()}`
380
+ : `Worksheet - ${new Date().toLocaleString()}`),
381
+ createdById: userId,
382
+ difficulty: difficultyUpper,
383
+ estimatedTime: input.estimatedTime,
384
+ generating: true,
385
+ generatingMetadata: {
386
+ quantity: resolved.numQuestions,
387
+ difficulty: resolved.difficulty,
388
+ mode: resolved.mode,
389
+ presetId: input.presetId ?? null,
390
+ },
391
+ worksheetConfig: worksheetConfigSnapshot,
392
+ },
393
+ });
394
+ await PusherService.emitTaskComplete(input.workspaceId, 'worksheet_info', {
395
+ contentLength: resolved.numQuestions,
396
+ });
397
+ await PusherService.emitWorksheetNew(input.workspaceId, artifact);
398
+ const workspaceId = input.workspaceId;
399
+ const promptText = input.prompt;
400
+ const estTime = input.estimatedTime;
401
+ (async () => {
402
+ try {
403
+ const content = await aiSessionService.generateWorksheetQuestions(workspaceId, userId, resolved.numQuestions, difficultyUpper, {
404
+ mode: resolved.mode,
405
+ mcqRatio: resolved.mcqRatio,
406
+ questionTypes: resolved.questionTypes,
407
+ prompt: promptText,
408
+ });
409
+ let problems = [];
410
+ try {
411
+ const worksheetData = JSON.parse(content);
412
+ let actualWorksheetData = worksheetData;
413
+ if (worksheetData.last_response) {
414
+ try {
415
+ actualWorksheetData = JSON.parse(worksheetData.last_response);
416
+ }
417
+ catch {
418
+ /* noop */
419
+ }
420
+ }
421
+ problems =
422
+ actualWorksheetData.problems ||
423
+ actualWorksheetData.questions ||
424
+ actualWorksheetData ||
425
+ [];
426
+ if (!Array.isArray(problems))
427
+ problems = [];
428
+ }
429
+ catch (parseError) {
430
+ this.logger.error('Failed to parse worksheet JSON', parseError);
431
+ throw new Error('Failed to parse worksheet JSON');
432
+ }
433
+ const forceMcq = resolved.mode === 'quiz';
434
+ const questionsToCreate = [];
435
+ for (let i = 0; i < Math.min(problems.length, resolved.numQuestions); i++) {
436
+ const problem = problems[i] && typeof problems[i] === 'object'
437
+ ? problems[i]
438
+ : {};
439
+ const row = normalizeWorksheetProblemForDb(problem, i, difficultyUpper, forceMcq);
440
+ const metaPayload = {};
441
+ if (row.meta.options?.length)
442
+ metaPayload.options = row.meta.options;
443
+ if (row.meta.markScheme != null)
444
+ metaPayload.markScheme = row.meta.markScheme;
445
+ questionsToCreate.push({
446
+ artifactId: artifact.id,
447
+ prompt: row.prompt,
448
+ answer: row.answer ?? '',
449
+ difficulty: row.difficulty,
450
+ order: row.order,
451
+ type: row.type,
452
+ meta: Object.keys(metaPayload).length ? metaPayload : undefined,
453
+ });
454
+ }
455
+ if (questionsToCreate.length > 0) {
456
+ await this.db.worksheetQuestion.createMany({ data: questionsToCreate });
457
+ }
458
+ let parsedTitle;
459
+ let parsedDescription;
460
+ let parsedEstimated;
461
+ try {
462
+ const worksheetData = JSON.parse(content);
463
+ let actualWorksheetData = worksheetData;
464
+ if (worksheetData.last_response) {
465
+ try {
466
+ actualWorksheetData = JSON.parse(worksheetData.last_response);
467
+ }
468
+ catch {
469
+ /* noop */
470
+ }
471
+ }
472
+ parsedTitle =
473
+ typeof actualWorksheetData.title === 'string' ? actualWorksheetData.title : undefined;
474
+ parsedDescription =
475
+ typeof actualWorksheetData.description === 'string'
476
+ ? actualWorksheetData.description
477
+ : undefined;
478
+ parsedEstimated =
479
+ typeof actualWorksheetData.estimatedTime === 'string'
480
+ ? actualWorksheetData.estimatedTime
481
+ : undefined;
482
+ }
483
+ catch {
484
+ /* noop */
485
+ }
486
+ await this.db.artifact.update({
487
+ where: { id: artifact.id },
488
+ data: {
489
+ generating: false,
490
+ title: parsedTitle || artifact.title,
491
+ description: parsedDescription ?? undefined,
492
+ estimatedTime: estTime || parsedEstimated || undefined,
493
+ worksheetConfig: worksheetConfigSnapshot,
494
+ },
495
+ });
496
+ const updatedWorksheet = await this.db.artifact.findFirst({
497
+ where: { id: artifact.id },
498
+ include: { questions: true },
499
+ });
500
+ await PusherService.emitWorksheetGenerationComplete(workspaceId, updatedWorksheet);
501
+ await PusherService.emitWorksheetComplete(workspaceId, artifact);
502
+ await notifyArtifactReady(this.db, {
503
+ userId,
504
+ workspaceId,
505
+ artifactId: artifact.id,
506
+ artifactType: ArtifactType.WORKSHEET,
507
+ title: updatedWorksheet?.title ?? artifact.title,
508
+ }).catch(() => { });
509
+ }
510
+ catch (error) {
511
+ this.logger.error(`Background generation failed for artifact ${artifact.id}:`, error);
512
+ await notifyArtifactFailed(this.db, {
513
+ userId,
514
+ workspaceId,
515
+ artifactId: artifact.id,
516
+ artifactType: ArtifactType.WORKSHEET,
517
+ title: artifact.title,
518
+ message: `Failed to generate worksheet: ${error instanceof Error ? error.message : 'Unknown error'}`,
519
+ }).catch(() => { });
520
+ await this.db.artifact.deleteMany({ where: { id: artifact.id } }).catch(() => { });
521
+ await PusherService.emitError(workspaceId, `Failed to generate worksheet: ${error instanceof Error ? error.message : 'Unknown error'}`, 'worksheet_generation');
522
+ }
523
+ })();
524
+ return { artifact };
525
+ }
526
+ async checkAnswer(userId, input) {
527
+ const worksheet = await this.db.artifact.findFirst({
528
+ where: {
529
+ id: input.worksheetId,
530
+ type: ArtifactType.WORKSHEET,
531
+ workspace: workspaceAccessWhere(userId),
532
+ },
533
+ include: { workspace: true },
534
+ });
535
+ if (!worksheet)
536
+ throw new TRPCError({ code: 'NOT_FOUND' });
537
+ const question = await this.db.worksheetQuestion.findFirst({
538
+ where: { id: input.questionId, artifactId: input.worksheetId },
539
+ });
540
+ if (!question)
541
+ throw new TRPCError({ code: 'NOT_FOUND' });
542
+ const questionMeta = question.meta
543
+ ? typeof question.meta === 'object'
544
+ ? question.meta
545
+ : JSON.parse(question.meta)
546
+ : {};
547
+ const markScheme = questionMeta.markScheme;
548
+ let isCorrect = false;
549
+ let userMarkScheme = null;
550
+ if (markScheme && markScheme.points && markScheme.points.length > 0) {
551
+ try {
552
+ userMarkScheme = await aiSessionService.checkWorksheetQuestions(worksheet.workspace.id, userId, question.prompt, input.answer, markScheme);
553
+ const achievedTotal = userMarkScheme.points.reduce((sum, p) => sum + (p.achievedPoints || 0), 0);
554
+ isCorrect = achievedTotal === markScheme.totalPoints;
555
+ }
556
+ catch (error) {
557
+ this.logger.error('Failed to check answer with AI', error instanceof Error ? error.message : 'Unknown error');
558
+ isCorrect = question.answer === input.answer;
559
+ }
560
+ }
561
+ else {
562
+ throw new TRPCError({ code: 'BAD_REQUEST', message: 'No mark scheme found for question' });
563
+ }
564
+ const progress = await this.db.worksheetQuestionProgress.upsert({
565
+ where: {
566
+ worksheetQuestionId_userId: {
567
+ worksheetQuestionId: input.questionId,
568
+ userId,
569
+ },
570
+ },
571
+ create: {
572
+ worksheetQuestionId: input.questionId,
573
+ userId,
574
+ modified: true,
575
+ userAnswer: input.answer,
576
+ correct: isCorrect,
577
+ completedAt: new Date(),
578
+ attempts: 1,
579
+ meta: userMarkScheme
580
+ ? { userMarkScheme: JSON.parse(JSON.stringify(userMarkScheme)) }
581
+ : { userMarkScheme: null },
582
+ },
583
+ update: {
584
+ modified: true,
585
+ userAnswer: input.answer,
586
+ correct: isCorrect,
587
+ completedAt: new Date(),
588
+ attempts: { increment: 1 },
589
+ meta: userMarkScheme
590
+ ? { userMarkScheme: JSON.parse(JSON.stringify(userMarkScheme)) }
591
+ : { userMarkScheme: null },
592
+ },
593
+ });
594
+ return { isCorrect, userMarkScheme, progress };
595
+ }
596
+ }
@@ -0,0 +1,24 @@
1
+ import type { PrismaClient } from '@prisma/client';
2
+ import { BaseService } from '../base.service.js';
3
+ /**
4
+ * Study analytics: streaks, weekly activity, flashcard mastery, worksheet
5
+ * accuracy. Behavior-preserving extraction from `workspace.getStudyAnalytics`.
6
+ */
7
+ export declare class WorkspaceAnalyticsService extends BaseService {
8
+ constructor(db: PrismaClient);
9
+ getStudyAnalytics(userId: string): Promise<{
10
+ streak: number;
11
+ totalStudyDays: number;
12
+ weeklyActivity: boolean[];
13
+ flashcards: {
14
+ total: number;
15
+ mastered: number;
16
+ dueForReview: number;
17
+ };
18
+ worksheets: {
19
+ completed: number;
20
+ correct: number;
21
+ accuracy: number;
22
+ };
23
+ }>;
24
+ }