@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
@@ -1,6 +1,6 @@
1
1
  import { z } from 'zod';
2
2
  import { authedProcedure, router } from '../trpc.js';
3
- import PusherService from '../lib/pusher.js';
3
+ import { NotificationService } from '../services/notifications/notification.service.js';
4
4
 
5
5
  const listInputSchema = z.object({
6
6
  cursor: z.string().optional(),
@@ -12,131 +12,29 @@ const listInputSchema = z.object({
12
12
  export const notifications = router({
13
13
  list: authedProcedure
14
14
  .input(listInputSchema)
15
- .query(async ({ ctx, input }) => {
16
- const { cursor, limit, unreadOnly, types } = input;
17
- const userId = ctx.userId;
15
+ .query(({ ctx, input }) => new NotificationService(ctx.db).list(ctx.userId, input)),
18
16
 
19
- const items = await ctx.db.notification.findMany({
20
- where: {
21
- userId,
22
- ...(unreadOnly ? { read: false } : {}),
23
- ...(types?.length ? { type: { in: types } } : {}),
24
- },
25
- include: {
26
- actor: {
27
- select: {
28
- id: true,
29
- name: true,
30
- email: true,
31
- },
32
- },
33
- workspace: {
34
- select: {
35
- id: true,
36
- title: true,
37
- },
38
- },
39
- },
40
- orderBy: [{ createdAt: 'desc' }, { id: 'desc' }],
41
- take: limit + 1,
42
- ...(cursor ? { cursor: { id: cursor }, skip: 1 } : {}),
43
- });
44
-
45
- let nextCursor: string | undefined;
46
- if (items.length > limit) {
47
- const next = items.pop();
48
- nextCursor = next?.id;
49
- }
50
-
51
- return {
52
- items,
53
- nextCursor,
54
- };
55
- }),
56
-
57
- unreadCount: authedProcedure.query(async ({ ctx }) => {
58
- const count = await ctx.db.notification.count({
59
- where: {
60
- userId: ctx.userId,
61
- read: false,
62
- },
63
- });
64
- return { count };
65
- }),
17
+ unreadCount: authedProcedure.query(({ ctx }) =>
18
+ new NotificationService(ctx.db).unreadCount(ctx.userId),
19
+ ),
66
20
 
67
21
  markRead: authedProcedure
68
22
  .input(z.object({ id: z.string() }))
69
- .mutation(async ({ ctx, input }) => {
70
- const now = new Date();
71
- await ctx.db.notification.updateMany({
72
- where: {
73
- id: input.id,
74
- userId: ctx.userId,
75
- },
76
- data: {
77
- read: true,
78
- readAt: now,
79
- },
80
- });
81
-
82
- const unreadCount = await ctx.db.notification.count({
83
- where: { userId: ctx.userId, read: false },
84
- });
85
- await PusherService.emitNotificationReadState(ctx.userId, { unreadCount });
86
-
87
- return { success: true };
88
- }),
23
+ .mutation(({ ctx, input }) =>
24
+ new NotificationService(ctx.db).markRead(ctx.userId, input.id),
25
+ ),
89
26
 
90
27
  markManyRead: authedProcedure
91
28
  .input(z.object({ ids: z.array(z.string()).min(1) }))
92
- .mutation(async ({ ctx, input }) => {
93
- const now = new Date();
94
- await ctx.db.notification.updateMany({
95
- where: {
96
- id: { in: input.ids },
97
- userId: ctx.userId,
98
- },
99
- data: {
100
- read: true,
101
- readAt: now,
102
- },
103
- });
104
-
105
- const unreadCount = await ctx.db.notification.count({
106
- where: { userId: ctx.userId, read: false },
107
- });
108
- await PusherService.emitNotificationReadState(ctx.userId, { unreadCount });
109
-
110
- return { success: true };
111
- }),
112
-
113
- markAllRead: authedProcedure
114
- .mutation(async ({ ctx }) => {
115
- const now = new Date();
116
- await ctx.db.notification.updateMany({
117
- where: {
118
- userId: ctx.userId,
119
- read: false,
120
- },
121
- data: {
122
- read: true,
123
- readAt: now,
124
- },
125
- });
29
+ .mutation(({ ctx, input }) =>
30
+ new NotificationService(ctx.db).markManyRead(ctx.userId, input.ids),
31
+ ),
126
32
 
127
- await PusherService.emitNotificationReadState(ctx.userId, { unreadCount: 0 });
128
- return { success: true };
129
- }),
33
+ markAllRead: authedProcedure.mutation(({ ctx }) =>
34
+ new NotificationService(ctx.db).markAllRead(ctx.userId),
35
+ ),
130
36
 
131
37
  delete: authedProcedure
132
38
  .input(z.object({ id: z.string() }))
133
- .mutation(async ({ ctx, input }) => {
134
- await ctx.db.notification.deleteMany({
135
- where: {
136
- id: input.id,
137
- userId: ctx.userId,
138
- },
139
- });
140
- return { success: true };
141
- }),
39
+ .mutation(({ ctx, input }) => new NotificationService(ctx.db).delete(ctx.userId, input.id)),
142
40
  });
@@ -1,450 +1,41 @@
1
1
  import { z } from 'zod';
2
- import { router, verifiedProcedure } from '../trpc.js';
3
- import { TRPCError } from '@trpc/server';
4
- import { stripe } from '../lib/stripe.js';
5
- import { env } from '../lib/env.js';
6
- import { getUserUsage, getUserPlanLimits } from '../lib/usage_service.js';
7
- import {
8
- notifyPaymentSucceeded,
9
- notifySubscriptionActivated,
10
- notifySubscriptionPaymentSucceeded,
11
- } from '../lib/notification-service.js';
12
- import { upsertSubscriptionFromStripe } from '../lib/subscription_service.js';
13
- import { ArtifactType } from '../lib/prisma.js';
14
- import { PrismaClient } from '@prisma/client';
15
- import { Stripe } from 'stripe';
16
-
17
- const ArtifactTypeUnion = z.enum(['STUDY_GUIDE', 'FLASHCARD_SET', 'WORKSHEET', 'MEETING_SUMMARY', 'PODCAST_EPISODE', 'STORAGE']);
18
-
19
- /** Stripe Checkout URLs contain the session id (cs_…); we only store the URL in DB. */
20
- const CHECKOUT_SESSION_ID_IN_URL = /cs_[a-zA-Z0-9]+/;
21
-
22
- /**
23
- * On idempotency conflict, we may have a stored checkout URL. Only reuse it if the
24
- * Stripe session is still `open`. Otherwise release the row so a new session can be created
25
- * (avoids redirecting users to a completed/expired checkout = "already completed").
26
- */
27
- async function reuseCheckoutUrlIfSessionStillOpen(
28
- stripeClient: Stripe,
29
- db: PrismaClient,
30
- existing: { id: string; stripeSessionId: string | null },
31
- lockKey: string,
32
- ): Promise<string | null> {
33
- const url = existing.stripeSessionId;
34
- if (!url) return null;
35
-
36
- const idMatch = url.match(CHECKOUT_SESSION_ID_IN_URL);
37
- if (!idMatch) {
38
- await db.idempotencyRecord.updateMany({
39
- where: { id: existing.id, activeLockKey: lockKey },
40
- data: { activeLockKey: null, status: 'expired' },
41
- });
42
- return null;
43
- }
44
-
45
- try {
46
- const session = await stripeClient.checkout.sessions.retrieve(idMatch[0]);
47
- if (session.status === 'open') {
48
- return url;
49
- }
50
- await db.idempotencyRecord.updateMany({
51
- where: { id: existing.id, activeLockKey: lockKey },
52
- data: {
53
- activeLockKey: null,
54
- status: session.status === 'complete' ? 'completed' : 'expired',
55
- },
56
- });
57
- return null;
58
- } catch {
59
- await db.idempotencyRecord.updateMany({
60
- where: { id: existing.id, activeLockKey: lockKey },
61
- data: { activeLockKey: null, status: 'expired' },
62
- });
63
- return null;
64
- }
65
- }
2
+ import { router, publicProcedure, verifiedProcedure } from '../trpc.js';
3
+ import { ArtifactType } from '@prisma/client';
4
+ import { PaymentService } from '../services/billing/payment.service.js';
66
5
 
67
6
  export const paymentRouter = router({
68
- getPlans: verifiedProcedure
69
- .query(async (opts) => {
70
- const ctx = opts.ctx;
71
- const userId = ctx.userId;
72
-
73
- const plans = await ctx.db.plan.findMany({
74
- where: { active: true },
75
- include: { limit: true },
76
- orderBy: { price: 'asc' }
77
- });
78
-
79
- const activeSubscriptions = await ctx.db.subscription.findMany({
80
- where: {
81
- userId: userId,
82
- status: 'active'
83
- }
84
- });
85
-
86
- return plans.map((plan: any) => {
87
- return {
88
- ...plan,
89
- isActive: activeSubscriptions.some((sub: any) => sub.planId === plan.id)
90
- };
91
- });
92
- }),
93
-
94
- createCheckoutSession: verifiedProcedure
95
- .input(z.object({
96
- planId: z.string()
97
- }))
98
- .mutation(async (opts) => {
99
- const ctx = opts.ctx;
100
- const input = opts.input;
101
- const userId = ctx.userId;
102
-
103
- if (!stripe) {
104
- throw new TRPCError({
105
- code: 'INTERNAL_SERVER_ERROR',
106
- message: 'Stripe not configured'
107
- });
108
- }
109
-
110
- const user = await ctx.db.user.findUnique({ where: { id: userId } });
111
- if (!user) throw new TRPCError({ code: 'NOT_FOUND', message: 'User not found' });
112
-
113
- const lockKey = `pending_${userId}_${input.planId}`;
114
- let attempt = null;
115
- let retryCount = 0;
116
-
117
- while (retryCount < 3) {
118
- try {
119
- attempt = await ctx.db.idempotencyRecord.create({
120
- data: {
121
- userId: userId,
122
- planId: input.planId,
123
- activeLockKey: lockKey,
124
- status: 'pending'
125
- }
126
- });
127
- break;
128
- } catch (err: any) {
129
- if (err.code === 'P2002') {
130
- const existing = await ctx.db.idempotencyRecord.findUnique({
131
- where: { activeLockKey: lockKey }
132
- });
133
- if (!existing) { retryCount++; continue; }
134
-
135
- const isStale = (Date.now() - existing.updatedAt.getTime()) > (24 * 60 * 60 * 1000);
136
- if (isStale) {
137
- const result = await ctx.db.idempotencyRecord.updateMany({
138
- where: { id: existing.id, activeLockKey: lockKey },
139
- data: { activeLockKey: null, status: 'expired' }
140
- });
141
- if (result.count > 0) { retryCount++; continue; }
142
- }
143
-
144
- if (existing.stripeSessionId && stripe) {
145
- const reusable = await reuseCheckoutUrlIfSessionStillOpen(
146
- stripe,
147
- ctx.db,
148
- existing,
149
- lockKey,
150
- );
151
- if (reusable) return { url: reusable };
152
- retryCount++;
153
- continue;
154
- }
155
- await new Promise(resolve => setTimeout(resolve, 800));
156
- retryCount++;
157
- continue;
158
- }
159
- throw err;
160
- }
161
- }
162
-
163
- if (!attempt) throw new TRPCError({ code: 'CONFLICT', message: "Concurrent request" });
164
-
165
- const plan = await ctx.db.plan.findUnique({ where: { id: input.planId } });
166
- if (!plan) throw new TRPCError({ code: 'NOT_FOUND', message: "Plan not found" });
167
-
168
- try {
169
- const successUrl = env.STRIPE_SUCCESS_URL.includes('session_id=')
170
- ? env.STRIPE_SUCCESS_URL
171
- : `${env.STRIPE_SUCCESS_URL}${env.STRIPE_SUCCESS_URL.includes('?') ? '&' : '?'}session_id={CHECKOUT_SESSION_ID}`;
172
-
173
- const session = await stripe.checkout.sessions.create({
174
- customer: user.stripe_customer_id || undefined,
175
- customer_email: user.stripe_customer_id ? undefined : (user.email || undefined),
176
- line_items: [{ price: plan.stripePriceId, quantity: 1 }],
177
- mode: plan.interval ? 'subscription' : 'payment',
178
- subscription_data: plan.interval ? {
179
- metadata: {
180
- userId: userId,
181
- planId: plan.id,
182
- attemptId: attempt.id
183
- }
184
- } : undefined,
185
- success_url: successUrl,
186
- cancel_url: env.STRIPE_CANCEL_URL,
187
- metadata: {
188
- userId: userId,
189
- planId: plan.id,
190
- attemptId: attempt.id
191
- }
192
- }, {
193
- idempotencyKey: attempt.id
194
- });
195
-
196
- await ctx.db.idempotencyRecord.update({
197
- where: { id: attempt.id },
198
- data: { stripeSessionId: session.url }
199
- });
200
-
201
- return { url: session.url };
202
- } catch (error: any) {
203
- await ctx.db.idempotencyRecord.update({
204
- where: { id: attempt.id },
205
- data: { status: 'failed', activeLockKey: null }
206
- });
207
- throw new TRPCError({
208
- code: 'INTERNAL_SERVER_ERROR',
209
- message: error.message
210
- });
211
- }
212
- }),
213
-
214
- confirmCheckoutSuccess: verifiedProcedure
215
- .input(z.object({
216
- sessionId: z.string().min(1),
217
- }))
218
- .mutation(async (opts) => {
219
- const ctx = opts.ctx;
220
- const input = opts.input;
221
-
222
- if (!stripe) {
223
- throw new TRPCError({
224
- code: 'INTERNAL_SERVER_ERROR',
225
- message: 'Stripe not configured'
226
- });
227
- }
228
-
229
- const session = await stripe.checkout.sessions.retrieve(input.sessionId, {
230
- expand: ['line_items', 'subscription'],
231
- });
232
-
233
- const metadata = session.metadata || {};
234
- if (metadata.userId !== ctx.userId) {
235
- throw new TRPCError({
236
- code: 'FORBIDDEN',
237
- message: 'This checkout session does not belong to the current user',
238
- });
239
- }
240
-
241
- if (session.status !== 'complete') {
242
- return { confirmed: false, reason: 'checkout_not_complete' };
243
- }
244
-
245
- const plan = metadata.planId
246
- ? await ctx.db.plan.findUnique({ where: { id: metadata.planId } })
247
- : null;
248
-
249
- if (session.mode === 'payment' && session.payment_status === 'paid') {
250
- await notifyPaymentSucceeded(ctx.db, {
251
- userId: ctx.userId,
252
- planId: metadata.planId,
253
- planName: plan?.name || metadata.planType,
254
- stripeSessionId: session.id,
255
- amountPaid: session.amount_total ?? undefined,
256
- });
257
- return { confirmed: true, kind: 'payment' as const };
258
- }
259
-
260
- if (session.mode === 'subscription') {
261
- const stripeSubscriptionId =
262
- typeof session.subscription === 'string'
263
- ? session.subscription
264
- : session.subscription?.id;
265
-
266
- if (stripeSubscriptionId) {
267
- await upsertSubscriptionFromStripe(stripeSubscriptionId);
268
- await notifySubscriptionActivated(ctx.db, {
269
- userId: ctx.userId,
270
- planId: metadata.planId,
271
- planName: plan?.name || metadata.planType,
272
- stripeSubscriptionId,
273
- });
274
-
275
- if (session.payment_status === 'paid') {
276
- await notifySubscriptionPaymentSucceeded(ctx.db, {
277
- userId: ctx.userId,
278
- planId: metadata.planId,
279
- planName: plan?.name || metadata.planType,
280
- stripeInvoiceId: `checkout_${session.id}`,
281
- amountPaid: session.amount_total ?? undefined,
282
- });
283
- }
284
- }
285
- return { confirmed: true, kind: 'subscription' as const };
286
- }
287
-
288
- return { confirmed: false, reason: 'unsupported_mode' };
289
- }),
290
-
291
- createResourcePurchaseSession: verifiedProcedure
292
- .input(z.object({
293
- resourceType: z.nativeEnum(ArtifactType),
294
- quantity: z.number().min(1).default(1),
295
- }))
296
- .mutation(async ({ ctx, input }) => {
297
- if (!stripe) {
298
- throw new TRPCError({
299
- code: 'INTERNAL_SERVER_ERROR',
300
- message: 'Stripe is not configured on the server',
301
- });
302
- }
303
-
304
- const userId = ctx.userId;
305
- const user = await ctx.db.user.findUnique({
306
- where: { id: userId },
307
- });
308
-
309
- if (!user) {
310
- throw new TRPCError({
311
- code: 'NOT_FOUND',
312
- message: 'User not found',
313
- });
314
- }
315
-
316
- const resourcePrice = await ctx.db.resourcePrice.findUnique({
317
- where: { resourceType: input.resourceType },
318
- });
319
-
320
- if (!resourcePrice) {
321
- throw new TRPCError({
322
- code: 'PRECONDITION_FAILED',
323
- message: 'Price not set',
324
- });
325
- }
326
-
327
- const lockKey = `topup_${userId}_${input.resourceType}`;
328
- let attempt = null;
329
- let retryCount = 0;
330
-
331
- while (retryCount < 3) {
332
- try {
333
- attempt = await ctx.db.idempotencyRecord.create({
334
- data: {
335
- userId: userId,
336
- resourceType: input.resourceType as ArtifactType,
337
- activeLockKey: lockKey,
338
- status: 'pending'
339
- }
340
- });
341
- break;
342
- } catch (err: any) {
343
- if (err.code === 'P2002') {
344
- const existing = await ctx.db.idempotencyRecord.findUnique({
345
- where: { activeLockKey: lockKey }
346
- });
347
- if (!existing) { retryCount++; continue; }
348
-
349
- const isStale = (Date.now() - existing.updatedAt.getTime()) > (24 * 60 * 60 * 1000);
350
- if (isStale) {
351
- const result = await ctx.db.idempotencyRecord.updateMany({
352
- where: { id: existing.id, activeLockKey: lockKey },
353
- data: { activeLockKey: null, status: 'expired' }
354
- });
355
- if (result.count > 0) { retryCount++; continue; }
356
- }
357
-
358
- if (existing.stripeSessionId && stripe) {
359
- const reusable = await reuseCheckoutUrlIfSessionStillOpen(
360
- stripe,
361
- ctx.db,
362
- existing,
363
- lockKey,
364
- );
365
- if (reusable) return { url: reusable };
366
- retryCount++;
367
- continue;
368
- }
369
- await new Promise(resolve => setTimeout(resolve, 800));
370
- retryCount++;
371
- continue;
372
- }
373
- throw err;
374
- }
375
- }
376
-
377
- if (!attempt) throw new TRPCError({ code: 'CONFLICT', message: "Concurrent request" });
378
-
379
- try {
380
- const resourceName = input.resourceType
381
- .split('_')
382
- .map((w) => w.charAt(0).toUpperCase() + w.slice(1).toLowerCase())
383
- .join(' ');
384
-
385
- const session = await stripe.checkout.sessions.create({
386
- customer: user.stripe_customer_id || undefined,
387
- line_items: [
388
- {
389
- price_data: {
390
- currency: 'usd',
391
- product_data: {
392
- name: `Add-on: extra ${resourceName}s`,
393
- description: `Purchase of ${input.quantity} additional ${resourceName}(s)`
394
- },
395
- unit_amount: resourcePrice.priceCents
396
- },
397
- quantity: input.quantity
398
- }
399
- ],
400
- mode: 'payment',
401
- success_url: `${env.STRIPE_SUCCESS_URL}?success=true`,
402
- cancel_url: env.STRIPE_CANCEL_URL,
403
- metadata: {
404
- userId: userId,
405
- resourceType: input.resourceType,
406
- quantity: input.quantity.toString(),
407
- isPurchase: 'true',
408
- attemptId: attempt.id,
409
- },
410
- invoice_creation: {
411
- enabled: true,
412
- },
413
- }, {
414
- idempotencyKey: attempt.id
415
- });
416
-
417
- await ctx.db.idempotencyRecord.update({
418
- where: { id: attempt.id },
419
- data: { stripeSessionId: session.url }
420
- });
421
-
422
- return { url: session.url };
423
- } catch (error: any) {
424
- await ctx.db.idempotencyRecord.update({
425
- where: { id: attempt.id },
426
- data: { status: 'failed', activeLockKey: null }
427
- });
428
- throw new TRPCError({
429
- code: 'INTERNAL_SERVER_ERROR',
430
- message: error.message
431
- });
432
- }
433
- }),
434
-
435
- getUsageOverview: verifiedProcedure
436
- .query(async (opts) => {
437
- const ctx = opts.ctx;
438
- const userId = ctx.userId;
439
- const [usage, limits] = await Promise.all([
440
- getUserUsage(userId),
441
- getUserPlanLimits(userId)
442
- ]);
443
- return { usage: usage, limits: limits, hasActivePlan: (!!limits) };
444
- }),
445
-
446
- getResourcePrices: verifiedProcedure
447
- .query(async (opts) => {
448
- return opts.ctx.db.resourcePrice.findMany();
449
- })
7
+ getPlans: publicProcedure.query(({ ctx }) =>
8
+ new PaymentService(ctx.db).getPlans(ctx.session?.user?.id),
9
+ ),
10
+
11
+ createCheckoutSession: verifiedProcedure
12
+ .input(z.object({ planId: z.string() }))
13
+ .mutation(({ ctx, input }) =>
14
+ new PaymentService(ctx.db).createCheckoutSession(ctx.userId, input),
15
+ ),
16
+
17
+ confirmCheckoutSuccess: verifiedProcedure
18
+ .input(z.object({ sessionId: z.string().min(1) }))
19
+ .mutation(({ ctx, input }) =>
20
+ new PaymentService(ctx.db).confirmCheckoutSuccess(ctx.userId, input),
21
+ ),
22
+
23
+ createResourcePurchaseSession: verifiedProcedure
24
+ .input(
25
+ z.object({
26
+ resourceType: z.nativeEnum(ArtifactType),
27
+ quantity: z.number().min(1).default(1),
28
+ }),
29
+ )
30
+ .mutation(({ ctx, input }) =>
31
+ new PaymentService(ctx.db).createResourcePurchaseSession(ctx.userId, input),
32
+ ),
33
+
34
+ getUsageOverview: verifiedProcedure.query(({ ctx }) =>
35
+ new PaymentService(ctx.db).getUsageOverview(ctx.userId),
36
+ ),
37
+
38
+ getResourcePrices: verifiedProcedure.query(({ ctx }) =>
39
+ new PaymentService(ctx.db).getResourcePrices(),
40
+ ),
450
41
  });