@goscribe/server 1.3.4 → 1.6.0

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