@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,184 @@
1
+ import { BaseService } from './base.service.js';
2
+ import { prisma } from '../lib/prisma.js';
3
+ import { DEFAULT_EMBEDDING_MODEL, embedQuery, embedTexts, toVectorLiteral, } from '../lib/embeddings.js';
4
+ import { chunkText } from '../lib/chunking.js';
5
+ import { WORKSPACE_KB_KIND } from '../lib/workspace-kb.js';
6
+ export class WorkspaceKbService extends BaseService {
7
+ constructor(db) {
8
+ super(db);
9
+ }
10
+ /** Returns the single auto-managed KB for a workspace, creating it if needed. */
11
+ async ensureWorkspaceKb(workspaceId, userId, title) {
12
+ const existingLink = await this.db.workspaceKnowledgeBase.findFirst({
13
+ where: {
14
+ workspaceId,
15
+ knowledgeBase: {
16
+ isCurated: false,
17
+ meta: {
18
+ path: ['kind'],
19
+ equals: WORKSPACE_KB_KIND,
20
+ },
21
+ },
22
+ },
23
+ include: { knowledgeBase: true },
24
+ });
25
+ if (existingLink) {
26
+ return existingLink.knowledgeBase;
27
+ }
28
+ const workspace = await this.db.workspace.findUnique({
29
+ where: { id: workspaceId },
30
+ select: { title: true },
31
+ });
32
+ const kb = await this.db.knowledgeBase.create({
33
+ data: {
34
+ createdById: userId,
35
+ name: `${title ?? workspace?.title ?? 'Workspace'} Knowledge Base`,
36
+ description: 'Private knowledge base for this workspace',
37
+ isCurated: false,
38
+ embeddingModel: DEFAULT_EMBEDDING_MODEL,
39
+ embeddingDim: 1536,
40
+ meta: { kind: WORKSPACE_KB_KIND, workspaceId },
41
+ },
42
+ });
43
+ await this.db.workspaceKnowledgeBase.create({
44
+ data: {
45
+ workspaceId,
46
+ knowledgeBaseId: kb.id,
47
+ addedById: userId,
48
+ },
49
+ });
50
+ return kb;
51
+ }
52
+ async getWorkspaceKbId(workspaceId) {
53
+ const link = await this.db.workspaceKnowledgeBase.findFirst({
54
+ where: {
55
+ workspaceId,
56
+ knowledgeBase: {
57
+ isCurated: false,
58
+ meta: { path: ['kind'], equals: WORKSPACE_KB_KIND },
59
+ },
60
+ },
61
+ select: { knowledgeBaseId: true },
62
+ });
63
+ return link?.knowledgeBaseId ?? null;
64
+ }
65
+ async indexText(workspaceId, userId, input) {
66
+ const trimmed = input.text.trim();
67
+ if (!trimmed)
68
+ return;
69
+ try {
70
+ const kb = await this.ensureWorkspaceKb(workspaceId, userId);
71
+ const docs = await this.db.knowledgeBaseDocument.findMany({
72
+ where: { knowledgeBaseId: kb.id },
73
+ select: { id: true, meta: true },
74
+ });
75
+ const existing = docs.find((doc) => {
76
+ const meta = doc.meta;
77
+ return meta?.sourceType === input.sourceType && meta?.sourceId === input.sourceId;
78
+ });
79
+ let documentId;
80
+ if (existing) {
81
+ documentId = existing.id;
82
+ await this.db.knowledgeBaseChunk.deleteMany({ where: { documentId } });
83
+ await this.db.knowledgeBaseDocument.update({
84
+ where: { id: documentId },
85
+ data: {
86
+ name: input.name,
87
+ status: 'PROCESSING',
88
+ errorMessage: null,
89
+ },
90
+ });
91
+ }
92
+ else {
93
+ const created = await this.db.knowledgeBaseDocument.create({
94
+ data: {
95
+ knowledgeBaseId: kb.id,
96
+ name: input.name,
97
+ mimeType: 'text/plain',
98
+ size: Buffer.byteLength(trimmed, 'utf8'),
99
+ status: 'PROCESSING',
100
+ meta: {
101
+ sourceType: input.sourceType,
102
+ sourceId: input.sourceId,
103
+ },
104
+ },
105
+ });
106
+ documentId = created.id;
107
+ }
108
+ await this.db.knowledgeBase.update({
109
+ where: { id: kb.id },
110
+ data: { status: 'INDEXING' },
111
+ });
112
+ const chunks = chunkText(trimmed, { targetTokens: 500, overlapTokens: 60 });
113
+ if (chunks.length === 0)
114
+ return;
115
+ const embeddings = await embedTexts(chunks.map((c) => c.content), { model: kb.embeddingModel });
116
+ for (let i = 0; i < chunks.length; i += 1) {
117
+ const chunk = chunks[i];
118
+ const vec = toVectorLiteral(embeddings[i] ?? []);
119
+ await this.db.$executeRawUnsafe(`INSERT INTO "KnowledgeBaseChunk"
120
+ ("id", "documentId", "knowledgeBaseId", "chunkIndex",
121
+ "pageNumber", "content", "tokenCount", "embedding", "createdAt")
122
+ VALUES (
123
+ gen_random_uuid()::text,
124
+ $1, $2, $3, $4, $5, $6, $7::vector, NOW()
125
+ )`, documentId, kb.id, chunk.index, null, chunk.content, chunk.approxTokens, vec);
126
+ }
127
+ await this.db.knowledgeBaseDocument.update({
128
+ where: { id: documentId },
129
+ data: { status: 'READY', numChunks: chunks.length },
130
+ });
131
+ await this.db.knowledgeBase.update({
132
+ where: { id: kb.id },
133
+ data: { status: 'READY' },
134
+ });
135
+ }
136
+ catch (err) {
137
+ this.logger.error(`Failed to index ${input.sourceType}:${input.sourceId} into workspace KB`, err);
138
+ }
139
+ }
140
+ /** Fire-and-forget indexing — never blocks the caller. */
141
+ indexTextAsync(workspaceId, userId, input) {
142
+ void this.indexText(workspaceId, userId, input);
143
+ }
144
+ async search(workspaceId, query, topK = 5) {
145
+ const kbId = await this.getWorkspaceKbId(workspaceId);
146
+ if (!kbId)
147
+ return [];
148
+ const kb = await this.db.knowledgeBase.findUnique({ where: { id: kbId } });
149
+ if (!kb)
150
+ return [];
151
+ const k = Math.min(Math.max(topK, 1), 20);
152
+ const queryVec = await embedQuery(query, { model: kb.embeddingModel });
153
+ const vecLiteral = toVectorLiteral(queryVec);
154
+ const rows = await this.db.$queryRawUnsafe(`SELECT c."id", c."documentId", c."chunkIndex", c."content",
155
+ (c."embedding" <=> $1::vector) AS "distance",
156
+ d."name" AS "documentName"
157
+ FROM "KnowledgeBaseChunk" c
158
+ JOIN "KnowledgeBaseDocument" d ON d."id" = c."documentId"
159
+ WHERE c."knowledgeBaseId" = $2
160
+ AND c."embedding" IS NOT NULL
161
+ ORDER BY c."embedding" <=> $1::vector
162
+ LIMIT ${k}`, vecLiteral, kbId);
163
+ return rows.map((r) => ({
164
+ chunkId: r.id,
165
+ documentId: r.documentId,
166
+ documentName: r.documentName,
167
+ chunkIndex: r.chunkIndex,
168
+ content: r.content,
169
+ score: 1 - Number(r.distance),
170
+ }));
171
+ }
172
+ formatContext(matches) {
173
+ if (matches.length === 0)
174
+ return '';
175
+ return matches
176
+ .map((m, i) => `[${i + 1}] (${m.documentName}, chunk ${m.chunkIndex})\n${m.content}`)
177
+ .join('\n\n---\n\n');
178
+ }
179
+ async retrieveContext(workspaceId, query, topK = 5) {
180
+ const matches = await this.search(workspaceId, query, topK);
181
+ return this.formatContext(matches);
182
+ }
183
+ }
184
+ export const workspaceKbService = new WorkspaceKbService(prisma);
@@ -0,0 +1,27 @@
1
+ import type { PrismaClient } from '@prisma/client';
2
+ export type StepStatus = 'pending' | 'in_progress' | 'completed' | 'skipped' | 'error';
3
+ export declare const PIPELINE_STEPS: readonly ["fileUpload", "fileAnalysis", "studyGuide", "flashcards"];
4
+ export declare const PROGRESS_FILENAME_MAX_CHARS = 20;
5
+ export declare function updateAnalysisProgress(db: PrismaClient, workspaceId: string, progress: any): Promise<void>;
6
+ export declare function truncateProgressFilename(value: string, maxChars?: number): string;
7
+ export declare function buildCombinedFilenameLabel(fileNames: string[]): string;
8
+ export declare function buildProgressSteps(currentStep: typeof PIPELINE_STEPS[number], currentStatus: StepStatus, config: {
9
+ generateStudyGuide: boolean;
10
+ generateFlashcards: boolean;
11
+ }, overrides?: Partial<Record<typeof PIPELINE_STEPS[number], StepStatus>>): Record<string, {
12
+ order: number;
13
+ status: StepStatus;
14
+ }>;
15
+ export declare function buildProgress(status: string, filename: string, fileType: string, currentStep: typeof PIPELINE_STEPS[number], currentStepStatus: StepStatus, config: {
16
+ generateStudyGuide: boolean;
17
+ generateFlashcards: boolean;
18
+ }, extra?: Record<string, any>): {
19
+ status: string;
20
+ filename: string;
21
+ fileType: string;
22
+ startedAt: string;
23
+ steps: Record<string, {
24
+ order: number;
25
+ status: StepStatus;
26
+ }>;
27
+ };
@@ -0,0 +1,56 @@
1
+ import PusherService from '../lib/pusher.js';
2
+ export const PIPELINE_STEPS = ['fileUpload', 'fileAnalysis', 'studyGuide', 'flashcards'];
3
+ export const PROGRESS_FILENAME_MAX_CHARS = 20;
4
+ export async function updateAnalysisProgress(db, workspaceId, progress) {
5
+ await db.workspace.update({
6
+ where: { id: workspaceId },
7
+ data: { analysisProgress: progress },
8
+ });
9
+ await PusherService.emitAnalysisProgress(workspaceId, progress);
10
+ }
11
+ export function truncateProgressFilename(value, maxChars = PROGRESS_FILENAME_MAX_CHARS) {
12
+ if (value.length <= maxChars)
13
+ return value;
14
+ return `${value.slice(0, Math.max(0, maxChars - 3))}...`;
15
+ }
16
+ export function buildCombinedFilenameLabel(fileNames) {
17
+ return truncateProgressFilename(fileNames.join(', '));
18
+ }
19
+ export function buildProgressSteps(currentStep, currentStatus, config, overrides) {
20
+ const stepIndex = PIPELINE_STEPS.indexOf(currentStep);
21
+ const steps = {};
22
+ for (let i = 0; i < PIPELINE_STEPS.length; i++) {
23
+ const step = PIPELINE_STEPS[i];
24
+ let status;
25
+ if (overrides?.[step]) {
26
+ status = overrides[step];
27
+ }
28
+ else if (i < stepIndex) {
29
+ status = 'completed';
30
+ }
31
+ else if (i === stepIndex) {
32
+ status = currentStatus;
33
+ }
34
+ else if (step === 'studyGuide' && !config.generateStudyGuide) {
35
+ status = 'skipped';
36
+ }
37
+ else if (step === 'flashcards' && !config.generateFlashcards) {
38
+ status = 'skipped';
39
+ }
40
+ else {
41
+ status = 'pending';
42
+ }
43
+ steps[step] = { order: i + 1, status };
44
+ }
45
+ return steps;
46
+ }
47
+ export function buildProgress(status, filename, fileType, currentStep, currentStepStatus, config, extra) {
48
+ return {
49
+ status,
50
+ filename,
51
+ fileType,
52
+ startedAt: new Date().toISOString(),
53
+ steps: buildProgressSteps(currentStep, currentStepStatus, config, extra),
54
+ ...extra,
55
+ };
56
+ }
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,49 @@
1
+ import { describe, it, expect } from 'vitest';
2
+ import { buildCombinedFilenameLabel, buildProgress, buildProgressSteps, truncateProgressFilename, } from './workspace-progress.service.js';
3
+ describe('truncateProgressFilename', () => {
4
+ it('returns value unchanged when under max', () => {
5
+ expect(truncateProgressFilename('hello.pdf')).toBe('hello.pdf');
6
+ });
7
+ it('truncates long values and appends ellipsis', () => {
8
+ const result = truncateProgressFilename('this-is-a-very-long-filename.pdf', 10);
9
+ expect(result).toBe('this-is...');
10
+ expect(result.length).toBe(10);
11
+ });
12
+ });
13
+ describe('buildCombinedFilenameLabel', () => {
14
+ it('joins filenames and truncates', () => {
15
+ expect(buildCombinedFilenameLabel(['a.pdf', 'b.pdf', 'c.pdf'])).toBe('a.pdf, b.pdf, c.pdf');
16
+ });
17
+ });
18
+ describe('buildProgressSteps', () => {
19
+ it('marks completed, current, and future correctly', () => {
20
+ const steps = buildProgressSteps('studyGuide', 'in_progress', {
21
+ generateStudyGuide: true,
22
+ generateFlashcards: true,
23
+ });
24
+ expect(steps.fileUpload.status).toBe('completed');
25
+ expect(steps.fileAnalysis.status).toBe('completed');
26
+ expect(steps.studyGuide.status).toBe('in_progress');
27
+ expect(steps.flashcards.status).toBe('pending');
28
+ });
29
+ it('respects generateStudyGuide and generateFlashcards flags', () => {
30
+ const steps = buildProgressSteps('fileUpload', 'in_progress', {
31
+ generateStudyGuide: false,
32
+ generateFlashcards: false,
33
+ });
34
+ expect(steps.studyGuide.status).toBe('skipped');
35
+ expect(steps.flashcards.status).toBe('skipped');
36
+ });
37
+ });
38
+ describe('buildProgress', () => {
39
+ it('returns serialized payload including steps', () => {
40
+ const payload = buildProgress('analyzing', 'doc.pdf', 'pdf', 'fileAnalysis', 'in_progress', {
41
+ generateStudyGuide: true,
42
+ generateFlashcards: true,
43
+ });
44
+ expect(payload.status).toBe('analyzing');
45
+ expect(payload.filename).toBe('doc.pdf');
46
+ expect(payload.fileType).toBe('pdf');
47
+ expect(payload.steps.fileAnalysis.status).toBe('in_progress');
48
+ });
49
+ });
@@ -0,0 +1,307 @@
1
+ import type { PrismaClient } from '@prisma/client';
2
+ import { BaseService } from './base.service.js';
3
+ export declare class WorkspaceService extends BaseService {
4
+ constructor(db: PrismaClient);
5
+ list(userId: string, parentId: string | null): Promise<{
6
+ workspaces: {
7
+ id: string;
8
+ createdAt: Date;
9
+ updatedAt: Date;
10
+ title: string;
11
+ description: string | null;
12
+ ownerId: string;
13
+ color: string;
14
+ markerColor: string | null;
15
+ icon: string;
16
+ folderId: string | null;
17
+ fileBeingAnalyzed: boolean;
18
+ analysisProgress: import("@prisma/client/runtime/library").JsonValue | null;
19
+ needsAnalysis: boolean;
20
+ }[];
21
+ folders: {
22
+ name: string;
23
+ id: string;
24
+ createdAt: Date;
25
+ updatedAt: Date;
26
+ ownerId: string;
27
+ parentId: string | null;
28
+ color: string;
29
+ markerColor: string | null;
30
+ }[];
31
+ }>;
32
+ getTree(userId: string): Promise<{
33
+ folders: {
34
+ name: string;
35
+ id: string;
36
+ createdAt: Date;
37
+ updatedAt: Date;
38
+ ownerId: string;
39
+ parentId: string | null;
40
+ color: string;
41
+ markerColor: string | null;
42
+ }[];
43
+ workspaces: ({
44
+ uploads: {
45
+ name: string;
46
+ id: string;
47
+ createdAt: Date;
48
+ mimeType: string;
49
+ }[];
50
+ } & {
51
+ id: string;
52
+ createdAt: Date;
53
+ updatedAt: Date;
54
+ title: string;
55
+ description: string | null;
56
+ ownerId: string;
57
+ color: string;
58
+ markerColor: string | null;
59
+ icon: string;
60
+ folderId: string | null;
61
+ fileBeingAnalyzed: boolean;
62
+ analysisProgress: import("@prisma/client/runtime/library").JsonValue | null;
63
+ needsAnalysis: boolean;
64
+ })[];
65
+ }>;
66
+ create(userId: string, input: {
67
+ name: string;
68
+ description?: string;
69
+ parentId?: string;
70
+ markerColor?: string | null;
71
+ }): Promise<{
72
+ id: string;
73
+ createdAt: Date;
74
+ updatedAt: Date;
75
+ title: string;
76
+ description: string | null;
77
+ ownerId: string;
78
+ color: string;
79
+ markerColor: string | null;
80
+ icon: string;
81
+ folderId: string | null;
82
+ fileBeingAnalyzed: boolean;
83
+ analysisProgress: import("@prisma/client/runtime/library").JsonValue | null;
84
+ needsAnalysis: boolean;
85
+ }>;
86
+ createFolder(userId: string, input: {
87
+ name: string;
88
+ color?: string;
89
+ parentId?: string;
90
+ }): Promise<{
91
+ name: string;
92
+ id: string;
93
+ createdAt: Date;
94
+ updatedAt: Date;
95
+ ownerId: string;
96
+ parentId: string | null;
97
+ color: string;
98
+ markerColor: string | null;
99
+ }>;
100
+ updateFolder(userId: string, input: {
101
+ id: string;
102
+ name?: string;
103
+ markerColor?: string | null;
104
+ }): Promise<{
105
+ name: string;
106
+ id: string;
107
+ createdAt: Date;
108
+ updatedAt: Date;
109
+ ownerId: string;
110
+ parentId: string | null;
111
+ color: string;
112
+ markerColor: string | null;
113
+ }>;
114
+ deleteFolder(userId: string, id: string): Promise<{
115
+ name: string;
116
+ id: string;
117
+ createdAt: Date;
118
+ updatedAt: Date;
119
+ ownerId: string;
120
+ parentId: string | null;
121
+ color: string;
122
+ markerColor: string | null;
123
+ }>;
124
+ get(userId: string, id: string): Promise<{
125
+ folder: {
126
+ name: string;
127
+ id: string;
128
+ createdAt: Date;
129
+ updatedAt: Date;
130
+ ownerId: string;
131
+ parentId: string | null;
132
+ color: string;
133
+ markerColor: string | null;
134
+ } | null;
135
+ artifacts: {
136
+ id: string;
137
+ createdAt: Date;
138
+ updatedAt: Date;
139
+ workspaceId: string;
140
+ type: import("@prisma/client").$Enums.ArtifactType;
141
+ title: string;
142
+ isArchived: boolean;
143
+ difficulty: import("@prisma/client").$Enums.Difficulty | null;
144
+ estimatedTime: string | null;
145
+ createdById: string | null;
146
+ description: string | null;
147
+ generating: boolean;
148
+ generatingMetadata: import("@prisma/client/runtime/library").JsonValue | null;
149
+ worksheetConfig: import("@prisma/client/runtime/library").JsonValue | null;
150
+ imageObjectKey: string | null;
151
+ }[];
152
+ uploads: {
153
+ meta: import("@prisma/client/runtime/library").JsonValue | null;
154
+ name: string;
155
+ id: string;
156
+ createdAt: Date;
157
+ workspaceId: string | null;
158
+ userId: string | null;
159
+ mimeType: string;
160
+ size: number;
161
+ bucket: string | null;
162
+ objectKey: string | null;
163
+ url: string | null;
164
+ checksum: string | null;
165
+ aiTranscription: import("@prisma/client/runtime/library").JsonValue | null;
166
+ }[];
167
+ } & {
168
+ id: string;
169
+ createdAt: Date;
170
+ updatedAt: Date;
171
+ title: string;
172
+ description: string | null;
173
+ ownerId: string;
174
+ color: string;
175
+ markerColor: string | null;
176
+ icon: string;
177
+ folderId: string | null;
178
+ fileBeingAnalyzed: boolean;
179
+ analysisProgress: import("@prisma/client/runtime/library").JsonValue | null;
180
+ needsAnalysis: boolean;
181
+ }>;
182
+ getStats(userId: string): Promise<{
183
+ workspaces: number;
184
+ folders: number;
185
+ lastUpdated: Date | undefined;
186
+ spaceUsed: number;
187
+ spaceTotal: number;
188
+ }>;
189
+ update(userId: string, input: {
190
+ id: string;
191
+ name?: string;
192
+ description?: string;
193
+ markerColor?: string | null;
194
+ icon?: string;
195
+ }): Promise<{
196
+ id: string;
197
+ createdAt: Date;
198
+ updatedAt: Date;
199
+ title: string;
200
+ description: string | null;
201
+ ownerId: string;
202
+ color: string;
203
+ markerColor: string | null;
204
+ icon: string;
205
+ folderId: string | null;
206
+ fileBeingAnalyzed: boolean;
207
+ analysisProgress: import("@prisma/client/runtime/library").JsonValue | null;
208
+ needsAnalysis: boolean;
209
+ }>;
210
+ delete(userId: string, id: string): Promise<boolean>;
211
+ getFolderInformation(userId: string, id: string): Promise<{
212
+ folder: {
213
+ name: string;
214
+ id: string;
215
+ createdAt: Date;
216
+ updatedAt: Date;
217
+ ownerId: string;
218
+ parentId: string | null;
219
+ color: string;
220
+ markerColor: string | null;
221
+ };
222
+ parents: {
223
+ name: string;
224
+ id: string;
225
+ createdAt: Date;
226
+ updatedAt: Date;
227
+ ownerId: string;
228
+ parentId: string | null;
229
+ color: string;
230
+ markerColor: string | null;
231
+ }[];
232
+ }>;
233
+ getSharedWith(userId: string, id: string): Promise<{
234
+ shared: {
235
+ id: string;
236
+ createdAt: Date;
237
+ updatedAt: Date;
238
+ title: string;
239
+ description: string | null;
240
+ ownerId: string;
241
+ color: string;
242
+ markerColor: string | null;
243
+ icon: string;
244
+ folderId: string | null;
245
+ fileBeingAnalyzed: boolean;
246
+ analysisProgress: import("@prisma/client/runtime/library").JsonValue | null;
247
+ needsAnalysis: boolean;
248
+ }[];
249
+ invitations: ({
250
+ workspace: {
251
+ id: string;
252
+ createdAt: Date;
253
+ updatedAt: Date;
254
+ title: string;
255
+ description: string | null;
256
+ ownerId: string;
257
+ color: string;
258
+ markerColor: string | null;
259
+ icon: string;
260
+ folderId: string | null;
261
+ fileBeingAnalyzed: boolean;
262
+ analysisProgress: import("@prisma/client/runtime/library").JsonValue | null;
263
+ needsAnalysis: boolean;
264
+ };
265
+ } & {
266
+ id: string;
267
+ email: string;
268
+ createdAt: Date;
269
+ updatedAt: Date;
270
+ workspaceId: string;
271
+ role: string;
272
+ token: string;
273
+ invitedById: string;
274
+ acceptedAt: Date | null;
275
+ expiresAt: Date;
276
+ })[];
277
+ }>;
278
+ uploadFiles(userId: string, input: {
279
+ id: string;
280
+ files: Array<{
281
+ filename: string;
282
+ contentType: string;
283
+ size: number;
284
+ }>;
285
+ }): Promise<{
286
+ fileId: string;
287
+ uploadUrl: string;
288
+ }[]>;
289
+ deleteFiles(userId: string, input: {
290
+ fileId: string[];
291
+ id: string;
292
+ }): Promise<boolean>;
293
+ getFileUploadUrl(userId: string, input: {
294
+ workspaceId: string;
295
+ filename: string;
296
+ contentType: string;
297
+ size: number;
298
+ }): Promise<{
299
+ fileId: string;
300
+ uploadUrl: string;
301
+ }>;
302
+ search(userId: string, input: {
303
+ query: string;
304
+ color?: string;
305
+ limit: number;
306
+ }): Promise<any[]>;
307
+ }