@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,556 @@
1
+ import type { PrismaClient } from '@prisma/client';
2
+ import { TRPCError } from '@trpc/server';
3
+ import { BaseService } from '../base.service.js';
4
+ import { ArtifactType } from '../../lib/constants.js';
5
+ import { supabaseClient } from '../../lib/storage.js';
6
+ import PusherService from '../../lib/pusher.js';
7
+ import { ai } from '../../lib/ai/index.js';
8
+ import { workspaceKbService } from '../workspace/workspace-kb.service.js';
9
+ import { getUserUsage, getUserPlanLimits } from '../billing/usage.service.js';
10
+ import { notifyArtifactFailed, notifyArtifactReady } from '../notifications/notification.service.js';
11
+
12
+ type StepStatus = 'pending' | 'in_progress' | 'completed' | 'skipped' | 'error';
13
+ const PIPELINE_STEPS = ['fileUpload', 'fileAnalysis', 'studyGuide', 'flashcards'] as const;
14
+ const PROGRESS_FILENAME_MAX_CHARS = 20;
15
+
16
+ function truncateProgressFilename(value: string, maxChars: number = PROGRESS_FILENAME_MAX_CHARS): string {
17
+ if (value.length <= maxChars) return value;
18
+ return `${value.slice(0, Math.max(0, maxChars - 3))}...`;
19
+ }
20
+
21
+ function buildCombinedFilenameLabel(fileNames: string[]): string {
22
+ return truncateProgressFilename(fileNames.join(', '));
23
+ }
24
+
25
+ function buildProgressSteps(
26
+ currentStep: typeof PIPELINE_STEPS[number],
27
+ currentStatus: StepStatus,
28
+ config: { generateStudyGuide: boolean; generateFlashcards: boolean },
29
+ overrides?: Partial<Record<typeof PIPELINE_STEPS[number], StepStatus>>,
30
+ ): Record<string, { order: number; status: StepStatus }> {
31
+ const stepIndex = PIPELINE_STEPS.indexOf(currentStep);
32
+ const steps: Record<string, { order: number; status: StepStatus }> = {};
33
+
34
+ for (let i = 0; i < PIPELINE_STEPS.length; i++) {
35
+ const step = PIPELINE_STEPS[i];
36
+ let status: StepStatus;
37
+
38
+ if (overrides?.[step]) {
39
+ status = overrides[step]!;
40
+ } else if (i < stepIndex) {
41
+ status = 'completed';
42
+ } else if (i === stepIndex) {
43
+ status = currentStatus;
44
+ } else {
45
+ if (step === 'studyGuide' && !config.generateStudyGuide) {
46
+ status = 'skipped';
47
+ } else if (step === 'flashcards' && !config.generateFlashcards) {
48
+ status = 'skipped';
49
+ } else {
50
+ status = 'pending';
51
+ }
52
+ }
53
+
54
+ steps[step] = { order: i + 1, status };
55
+ }
56
+
57
+ return steps;
58
+ }
59
+
60
+ function buildProgress(
61
+ status: string,
62
+ filename: string,
63
+ fileType: string,
64
+ currentStep: typeof PIPELINE_STEPS[number],
65
+ currentStepStatus: StepStatus,
66
+ config: { generateStudyGuide: boolean; generateFlashcards: boolean },
67
+ extra?: Record<string, any>,
68
+ ) {
69
+ return {
70
+ status,
71
+ filename,
72
+ fileType,
73
+ startedAt: new Date().toISOString(),
74
+ steps: buildProgressSteps(currentStep, currentStepStatus, config, extra as any),
75
+ ...extra,
76
+ };
77
+ }
78
+
79
+ export interface UploadAndAnalyzeInput {
80
+ workspaceId: string;
81
+ files: Array<{ id: string }>;
82
+ generateStudyGuide: boolean;
83
+ generateFlashcards: boolean;
84
+ generateWorksheet: boolean;
85
+ }
86
+
87
+ export class MediaAnalysisService extends BaseService {
88
+ constructor(db: PrismaClient) {
89
+ super(db);
90
+ }
91
+
92
+ private async updateAnalysisProgress(workspaceId: string, progress: any) {
93
+ await this.db.workspace.update({
94
+ where: { id: workspaceId },
95
+ data: { analysisProgress: progress },
96
+ });
97
+ await PusherService.emitAnalysisProgress(workspaceId, progress);
98
+ }
99
+
100
+ async uploadAndAnalyzeMedia(input: UploadAndAnalyzeInput, userId: string) {
101
+ const workspace = await this.db.workspace.findFirst({
102
+ where: { id: input.workspaceId, ownerId: userId },
103
+ });
104
+ if (!workspace) {
105
+ this.logger.error('Workspace not found', { workspaceId: input.workspaceId, userId });
106
+ throw new TRPCError({ code: 'NOT_FOUND' });
107
+ }
108
+
109
+ if (workspace.fileBeingAnalyzed) {
110
+ throw new TRPCError({
111
+ code: 'CONFLICT',
112
+ message:
113
+ 'File analysis is already in progress for this workspace. Please wait for it to complete.',
114
+ });
115
+ }
116
+
117
+ const files = await this.db.fileAsset.findMany({
118
+ where: {
119
+ id: { in: input.files.map((file) => file.id) },
120
+ workspaceId: input.workspaceId,
121
+ userId,
122
+ },
123
+ });
124
+
125
+ if (files.length === 0) {
126
+ throw new TRPCError({
127
+ code: 'NOT_FOUND',
128
+ message: 'No files found with the provided IDs',
129
+ });
130
+ }
131
+
132
+ for (const file of files) {
133
+ if (!file.bucket || !file.objectKey) {
134
+ throw new TRPCError({
135
+ code: 'BAD_REQUEST',
136
+ message: `File ${file.id} does not have bucket or objectKey set`,
137
+ });
138
+ }
139
+ }
140
+
141
+ const primaryFile = files[0];
142
+ const fileType = primaryFile.mimeType.startsWith('image/') ? 'image' : 'pdf';
143
+ const combinedFilenameLabel = buildCombinedFilenameLabel(files.map((file) => file.name));
144
+
145
+ try {
146
+ await this.db.workspace.update({
147
+ where: { id: input.workspaceId },
148
+ data: { fileBeingAnalyzed: true },
149
+ });
150
+
151
+ const genConfig = {
152
+ generateStudyGuide: input.generateStudyGuide,
153
+ generateFlashcards: input.generateFlashcards,
154
+ };
155
+
156
+ PusherService.emitAnalysisProgress(
157
+ input.workspaceId,
158
+ buildProgress('starting', primaryFile.name, fileType, 'fileUpload', 'pending', genConfig),
159
+ );
160
+
161
+ try {
162
+ await this.updateAnalysisProgress(
163
+ input.workspaceId,
164
+ buildProgress('starting', primaryFile.name, fileType, 'fileUpload', 'pending', genConfig),
165
+ );
166
+ } catch (error) {
167
+ this.logger.error('Failed to update analysis progress:', error);
168
+ await this.db.workspace.update({
169
+ where: { id: input.workspaceId },
170
+ data: { fileBeingAnalyzed: false },
171
+ });
172
+ await PusherService.emitError(
173
+ input.workspaceId,
174
+ `Failed to update analysis progress: ${error}`,
175
+ 'file_analysis',
176
+ );
177
+ throw error;
178
+ }
179
+
180
+ await this.updateAnalysisProgress(
181
+ input.workspaceId,
182
+ buildProgress('uploading', primaryFile.name, fileType, 'fileUpload', 'in_progress', genConfig),
183
+ );
184
+
185
+ for (const [fileIndex, file] of files.entries()) {
186
+ if (!file.bucket || !file.objectKey) {
187
+ continue;
188
+ }
189
+
190
+ const currentFileType = file.mimeType.startsWith('image/') ? 'image' : 'pdf';
191
+ const currentFileLabel = truncateProgressFilename(file.name);
192
+
193
+ await this.updateAnalysisProgress(
194
+ input.workspaceId,
195
+ buildProgress('uploading', currentFileLabel, currentFileType, 'fileUpload', 'in_progress', genConfig, {
196
+ currentFileIndex: fileIndex + 1,
197
+ totalFiles: files.length,
198
+ }),
199
+ );
200
+
201
+ const { data: signedUrlData, error: signedUrlError } = await supabaseClient.storage
202
+ .from(file.bucket)
203
+ .createSignedUrl(file.objectKey, 24 * 60 * 60);
204
+
205
+ if (signedUrlError) {
206
+ await this.db.workspace.update({
207
+ where: { id: input.workspaceId },
208
+ data: { fileBeingAnalyzed: false },
209
+ });
210
+ throw new TRPCError({
211
+ code: 'INTERNAL_SERVER_ERROR',
212
+ message: `Failed to upload file`,
213
+ });
214
+ }
215
+
216
+ const fileUrl = signedUrlData.signedUrl;
217
+ const maxPages = currentFileType === 'pdf' && file.size && file.size > 50 ? 50 : undefined;
218
+
219
+ const processResult = await ai.backend.processFile(
220
+ input.workspaceId,
221
+ userId,
222
+ fileUrl,
223
+ currentFileType,
224
+ maxPages,
225
+ );
226
+
227
+ if (processResult.status === 'error') {
228
+ this.logger.error(`Failed to process file ${file.name}:`, processResult.error);
229
+ } else {
230
+ this.logger.info(
231
+ `Successfully processed file ${file.name}: ${processResult.pageCount} pages`,
232
+ );
233
+
234
+ await this.db.fileAsset.update({
235
+ where: { id: file.id },
236
+ data: {
237
+ aiTranscription: {
238
+ comprehensiveDescription: processResult.comprehensiveDescription,
239
+ textContent: processResult.textContent,
240
+ imageDescriptions: processResult.imageDescriptions,
241
+ },
242
+ },
243
+ });
244
+
245
+ const indexText =
246
+ processResult.comprehensiveDescription ??
247
+ processResult.textContent ??
248
+ '';
249
+ if (indexText.trim()) {
250
+ workspaceKbService.indexTextAsync(input.workspaceId, userId, {
251
+ sourceType: 'file',
252
+ sourceId: file.id,
253
+ name: file.name,
254
+ text: indexText,
255
+ });
256
+ }
257
+ }
258
+ }
259
+
260
+ await this.updateAnalysisProgress(
261
+ input.workspaceId,
262
+ buildProgress('analyzing', combinedFilenameLabel, fileType, 'fileAnalysis', 'in_progress', genConfig),
263
+ );
264
+
265
+ try {
266
+ await this.updateAnalysisProgress(
267
+ input.workspaceId,
268
+ buildProgress('generating_artifacts', combinedFilenameLabel, fileType, 'studyGuide', 'pending', genConfig),
269
+ );
270
+ } catch (error) {
271
+ this.logger.error('Failed to analyze files:', error);
272
+ await this.updateAnalysisProgress(
273
+ input.workspaceId,
274
+ buildProgress('error', combinedFilenameLabel, fileType, 'fileAnalysis', 'error', genConfig, {
275
+ error: `Failed to analyze ${fileType}: ${error}`,
276
+ studyGuide: 'skipped',
277
+ flashcards: 'skipped',
278
+ }),
279
+ );
280
+ await this.db.workspace.update({
281
+ where: { id: input.workspaceId },
282
+ data: { fileBeingAnalyzed: false },
283
+ });
284
+ throw error;
285
+ }
286
+
287
+ const results: {
288
+ filename: string;
289
+ artifacts: {
290
+ studyGuide: any | null;
291
+ flashcards: any | null;
292
+ worksheet: any | null;
293
+ };
294
+ } = {
295
+ filename: primaryFile.name,
296
+ artifacts: {
297
+ studyGuide: null,
298
+ flashcards: null,
299
+ worksheet: null,
300
+ },
301
+ };
302
+
303
+ try {
304
+ await ai.backend.initSession(input.workspaceId, userId);
305
+ } catch (initError) {
306
+ this.logger.error('Failed to init AI session (continuing with workspace context):', initError);
307
+ }
308
+
309
+ const [usage, limits] = await Promise.all([
310
+ getUserUsage(userId),
311
+ getUserPlanLimits(userId),
312
+ ]);
313
+
314
+ if (input.generateStudyGuide) {
315
+ if (limits && usage.studyGuides >= limits.maxStudyGuides) {
316
+ await this.updateAnalysisProgress(
317
+ input.workspaceId,
318
+ buildProgress('skipped', combinedFilenameLabel, fileType, 'studyGuide', 'skipped', genConfig),
319
+ );
320
+ await PusherService.emitError(input.workspaceId, 'Study guide skipped: Limit reached.', 'study_guide');
321
+ await notifyArtifactFailed(this.db, {
322
+ userId,
323
+ workspaceId: input.workspaceId,
324
+ artifactType: ArtifactType.STUDY_GUIDE,
325
+ message: 'Study guide was skipped because your plan limit was reached.',
326
+ }).catch(() => {});
327
+ } else {
328
+ try {
329
+ await this.updateAnalysisProgress(
330
+ input.workspaceId,
331
+ buildProgress('generating_study_guide', combinedFilenameLabel, fileType, 'studyGuide', 'in_progress', genConfig),
332
+ );
333
+
334
+ const ragContext = await workspaceKbService.retrieveContext(
335
+ input.workspaceId,
336
+ 'study guide key concepts definitions explanations examples',
337
+ 8,
338
+ );
339
+
340
+ const content = await ai.backend.generateStudyGuide(
341
+ input.workspaceId,
342
+ userId,
343
+ { ragContext },
344
+ );
345
+
346
+ let artifact = await this.db.artifact.findFirst({
347
+ where: { workspaceId: input.workspaceId, type: ArtifactType.STUDY_GUIDE },
348
+ });
349
+ if (!artifact) {
350
+ artifact = await this.db.artifact.create({
351
+ data: {
352
+ workspaceId: input.workspaceId,
353
+ type: ArtifactType.STUDY_GUIDE,
354
+ title:
355
+ files.length === 1
356
+ ? `Study Guide - ${primaryFile.name}`
357
+ : `Study Guide - ${files.length} files`,
358
+ createdById: userId,
359
+ },
360
+ });
361
+ }
362
+
363
+ const lastVersion = await this.db.artifactVersion.findFirst({
364
+ where: { artifact: { workspaceId: input.workspaceId, type: ArtifactType.STUDY_GUIDE } },
365
+ orderBy: { version: 'desc' },
366
+ });
367
+
368
+ await this.db.artifactVersion.create({
369
+ data: {
370
+ artifactId: artifact.id,
371
+ version: lastVersion ? lastVersion.version + 1 : 1,
372
+ content: content,
373
+ createdById: userId,
374
+ },
375
+ });
376
+
377
+ results.artifacts.studyGuide = artifact;
378
+ await PusherService.emitStudyGuideComplete(input.workspaceId, artifact);
379
+ await notifyArtifactReady(this.db, {
380
+ userId,
381
+ workspaceId: input.workspaceId,
382
+ artifactId: artifact.id,
383
+ artifactType: ArtifactType.STUDY_GUIDE,
384
+ title: artifact.title,
385
+ }).catch(() => {});
386
+
387
+ workspaceKbService.indexTextAsync(input.workspaceId, userId, {
388
+ sourceType: 'study-guide',
389
+ sourceId: artifact.id,
390
+ name: artifact.title,
391
+ text: content,
392
+ });
393
+ } catch (sgError) {
394
+ this.logger.error('Study guide generation failed after retries:', sgError);
395
+ await PusherService.emitError(
396
+ input.workspaceId,
397
+ 'Study guide generation failed. Please try regenerating later.',
398
+ 'study_guide',
399
+ );
400
+ await notifyArtifactFailed(this.db, {
401
+ userId,
402
+ workspaceId: input.workspaceId,
403
+ artifactType: ArtifactType.STUDY_GUIDE,
404
+ message: 'Study guide generation failed. Please try regenerating later.',
405
+ }).catch(() => {});
406
+ }
407
+ }
408
+ }
409
+
410
+ if (input.generateFlashcards) {
411
+ if (limits && usage.flashcards >= limits.maxFlashcards) {
412
+ await this.updateAnalysisProgress(
413
+ input.workspaceId,
414
+ buildProgress('skipped', combinedFilenameLabel, fileType, 'flashcards', 'skipped', genConfig),
415
+ );
416
+ await PusherService.emitError(input.workspaceId, 'Flashcards skipped: Limit reached.', 'flashcards');
417
+ await notifyArtifactFailed(this.db, {
418
+ userId,
419
+ workspaceId: input.workspaceId,
420
+ artifactType: ArtifactType.FLASHCARD_SET,
421
+ message: 'Flashcards were skipped because your plan limit was reached.',
422
+ }).catch(() => {});
423
+ } else {
424
+ try {
425
+ const sgStatus = input.generateStudyGuide
426
+ ? results.artifacts.studyGuide
427
+ ? 'completed'
428
+ : 'error'
429
+ : 'skipped';
430
+ await this.updateAnalysisProgress(
431
+ input.workspaceId,
432
+ buildProgress('generating_flashcards', combinedFilenameLabel, fileType, 'flashcards', 'in_progress', genConfig, {
433
+ studyGuide: sgStatus,
434
+ } as any),
435
+ );
436
+
437
+ const flashcardRag = await workspaceKbService.retrieveContext(
438
+ input.workspaceId,
439
+ 'flashcard key concepts facts definitions questions',
440
+ 8,
441
+ );
442
+
443
+ const content = await ai.backend.generateFlashcardQuestions(
444
+ input.workspaceId,
445
+ userId,
446
+ 10,
447
+ 'medium',
448
+ undefined,
449
+ { ragContext: flashcardRag },
450
+ );
451
+
452
+ const artifact = await this.db.artifact.create({
453
+ data: {
454
+ workspaceId: input.workspaceId,
455
+ type: ArtifactType.FLASHCARD_SET,
456
+ title:
457
+ files.length === 1
458
+ ? `Flashcards - ${primaryFile.name}`
459
+ : `Flashcards - ${files.length} files`,
460
+ createdById: userId,
461
+ },
462
+ });
463
+
464
+ try {
465
+ const parsed = typeof content === 'string' ? JSON.parse(content) : content;
466
+ const flashcardData = Array.isArray(parsed) ? parsed : (parsed.flashcards || []);
467
+
468
+ for (let i = 0; i < Math.min(flashcardData.length, 10); i++) {
469
+ const card = flashcardData[i];
470
+ const front = card.term || card.front || card.question || card.prompt || `Question ${i + 1}`;
471
+ const back = card.definition || card.back || card.answer || card.solution || `Answer ${i + 1}`;
472
+
473
+ await this.db.flashcard.create({
474
+ data: {
475
+ artifactId: artifact.id,
476
+ front: front,
477
+ back: back,
478
+ order: i,
479
+ tags: ['ai-generated', 'medium'],
480
+ },
481
+ });
482
+ }
483
+ } catch (parseError) {
484
+ console.error(
485
+ 'Failed to parse flashcard JSON or create cards in workspace router:',
486
+ parseError,
487
+ );
488
+ const lines = content.split('\n').filter((line: string) => line.trim());
489
+ for (let i = 0; i < Math.min(lines.length, 10); i++) {
490
+ const line = lines[i];
491
+ if (line.includes(' - ')) {
492
+ const [front, back] = line.split(' - ');
493
+ await this.db.flashcard.create({
494
+ data: {
495
+ artifactId: artifact.id,
496
+ front: front.trim(),
497
+ back: back.trim(),
498
+ order: i,
499
+ tags: ['ai-generated', 'medium'],
500
+ },
501
+ });
502
+ }
503
+ }
504
+ }
505
+
506
+ results.artifacts.flashcards = artifact;
507
+ await PusherService.emitFlashcardComplete(input.workspaceId, artifact);
508
+ await notifyArtifactReady(this.db, {
509
+ userId,
510
+ workspaceId: input.workspaceId,
511
+ artifactId: artifact.id,
512
+ artifactType: ArtifactType.FLASHCARD_SET,
513
+ title: artifact.title,
514
+ }).catch(() => {});
515
+ } catch (fcError) {
516
+ this.logger.error('Flashcard generation failed after retries:', fcError);
517
+ await PusherService.emitError(
518
+ input.workspaceId,
519
+ 'Flashcard generation failed. Please try regenerating later.',
520
+ 'flashcards',
521
+ );
522
+ await notifyArtifactFailed(this.db, {
523
+ userId,
524
+ workspaceId: input.workspaceId,
525
+ artifactType: ArtifactType.FLASHCARD_SET,
526
+ message: 'Flashcard generation failed. Please try regenerating later.',
527
+ }).catch(() => {});
528
+ }
529
+ }
530
+ }
531
+
532
+ await this.db.workspace.update({
533
+ where: { id: input.workspaceId },
534
+ data: { fileBeingAnalyzed: false },
535
+ });
536
+
537
+ await this.updateAnalysisProgress(input.workspaceId, {
538
+ ...buildProgress('completed', combinedFilenameLabel, fileType, 'flashcards', 'completed', genConfig),
539
+ completedAt: new Date().toISOString(),
540
+ });
541
+ return results;
542
+ } catch (error) {
543
+ this.logger.error('Failed to update analysis progress:', error);
544
+ await this.db.workspace.update({
545
+ where: { id: input.workspaceId },
546
+ data: { fileBeingAnalyzed: false },
547
+ });
548
+ await PusherService.emitError(
549
+ input.workspaceId,
550
+ `Failed to update analysis progress: ${error}`,
551
+ 'file_analysis',
552
+ );
553
+ throw error;
554
+ }
555
+ }
556
+ }