@goscribe/server 1.5.0 → 1.7.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 (321) hide show
  1. package/dist/context.d.ts +14 -1
  2. package/dist/context.js +23 -2
  3. package/dist/generated/prisma/client.d.ts +224 -0
  4. package/dist/generated/prisma/client.js +34 -0
  5. package/dist/generated/prisma/commonInputTypes.d.ts +941 -0
  6. package/dist/generated/prisma/commonInputTypes.js +10 -0
  7. package/dist/generated/prisma/enums.d.ts +67 -0
  8. package/dist/generated/prisma/enums.js +66 -0
  9. package/dist/generated/prisma/internal/class.d.ts +539 -0
  10. package/dist/generated/prisma/internal/class.js +49 -0
  11. package/dist/generated/prisma/internal/prismaNamespace.d.ts +3924 -0
  12. package/dist/generated/prisma/internal/prismaNamespace.js +557 -0
  13. package/dist/generated/prisma/models/ActivityLog.d.ts +1847 -0
  14. package/dist/generated/prisma/models/ActivityLog.js +1 -0
  15. package/dist/generated/prisma/models/Artifact.d.ts +2345 -0
  16. package/dist/generated/prisma/models/Artifact.js +1 -0
  17. package/dist/generated/prisma/models/ArtifactVersion.d.ts +1550 -0
  18. package/dist/generated/prisma/models/ArtifactVersion.js +1 -0
  19. package/dist/generated/prisma/models/Channel.d.ts +1257 -0
  20. package/dist/generated/prisma/models/Channel.js +1 -0
  21. package/dist/generated/prisma/models/Chat.d.ts +1339 -0
  22. package/dist/generated/prisma/models/Chat.js +1 -0
  23. package/dist/generated/prisma/models/CopilotConversation.d.ts +1450 -0
  24. package/dist/generated/prisma/models/CopilotConversation.js +1 -0
  25. package/dist/generated/prisma/models/CopilotMessage.d.ts +1179 -0
  26. package/dist/generated/prisma/models/CopilotMessage.js +1 -0
  27. package/dist/generated/prisma/models/FileAsset.d.ts +1832 -0
  28. package/dist/generated/prisma/models/FileAsset.js +1 -0
  29. package/dist/generated/prisma/models/Flashcard.d.ts +1460 -0
  30. package/dist/generated/prisma/models/Flashcard.js +1 -0
  31. package/dist/generated/prisma/models/FlashcardProgress.d.ts +1782 -0
  32. package/dist/generated/prisma/models/FlashcardProgress.js +1 -0
  33. package/dist/generated/prisma/models/Folder.d.ts +1685 -0
  34. package/dist/generated/prisma/models/Folder.js +1 -0
  35. package/dist/generated/prisma/models/IdempotencyRecord.d.ts +1319 -0
  36. package/dist/generated/prisma/models/IdempotencyRecord.js +1 -0
  37. package/dist/generated/prisma/models/Invoice.d.ts +1586 -0
  38. package/dist/generated/prisma/models/Invoice.js +1 -0
  39. package/dist/generated/prisma/models/KnowledgeBase.d.ts +1721 -0
  40. package/dist/generated/prisma/models/KnowledgeBase.js +1 -0
  41. package/dist/generated/prisma/models/KnowledgeBaseChunk.d.ts +1333 -0
  42. package/dist/generated/prisma/models/KnowledgeBaseChunk.js +1 -0
  43. package/dist/generated/prisma/models/KnowledgeBaseDocument.d.ts +1695 -0
  44. package/dist/generated/prisma/models/KnowledgeBaseDocument.js +1 -0
  45. package/dist/generated/prisma/models/Notification.d.ts +1992 -0
  46. package/dist/generated/prisma/models/Notification.js +1 -0
  47. package/dist/generated/prisma/models/PasswordResetToken.d.ts +1210 -0
  48. package/dist/generated/prisma/models/PasswordResetToken.js +1 -0
  49. package/dist/generated/prisma/models/Plan.d.ts +1431 -0
  50. package/dist/generated/prisma/models/Plan.js +1 -0
  51. package/dist/generated/prisma/models/PlanLimit.d.ts +1328 -0
  52. package/dist/generated/prisma/models/PlanLimit.js +1 -0
  53. package/dist/generated/prisma/models/PodcastSegment.d.ts +1564 -0
  54. package/dist/generated/prisma/models/PodcastSegment.js +1 -0
  55. package/dist/generated/prisma/models/ResourcePrice.d.ts +1008 -0
  56. package/dist/generated/prisma/models/ResourcePrice.js +1 -0
  57. package/dist/generated/prisma/models/Role.d.ts +1065 -0
  58. package/dist/generated/prisma/models/Role.js +1 -0
  59. package/dist/generated/prisma/models/Session.d.ts +1105 -0
  60. package/dist/generated/prisma/models/Session.js +1 -0
  61. package/dist/generated/prisma/models/StripeEvent.d.ts +1081 -0
  62. package/dist/generated/prisma/models/StripeEvent.js +1 -0
  63. package/dist/generated/prisma/models/StudyGuideComment.d.ts +1321 -0
  64. package/dist/generated/prisma/models/StudyGuideComment.js +1 -0
  65. package/dist/generated/prisma/models/StudyGuideHighlight.d.ts +1629 -0
  66. package/dist/generated/prisma/models/StudyGuideHighlight.js +1 -0
  67. package/dist/generated/prisma/models/Subscription.d.ts +1677 -0
  68. package/dist/generated/prisma/models/Subscription.js +1 -0
  69. package/dist/generated/prisma/models/User.d.ts +7559 -0
  70. package/dist/generated/prisma/models/User.js +1 -0
  71. package/dist/generated/prisma/models/UserCredit.d.ts +1249 -0
  72. package/dist/generated/prisma/models/UserCredit.js +1 -0
  73. package/dist/generated/prisma/models/VerificationToken.d.ts +946 -0
  74. package/dist/generated/prisma/models/VerificationToken.js +1 -0
  75. package/dist/generated/prisma/models/WorksheetPreset.d.ts +1433 -0
  76. package/dist/generated/prisma/models/WorksheetPreset.js +1 -0
  77. package/dist/generated/prisma/models/WorksheetQuestion.d.ts +1491 -0
  78. package/dist/generated/prisma/models/WorksheetQuestion.js +1 -0
  79. package/dist/generated/prisma/models/WorksheetQuestionProgress.d.ts +1620 -0
  80. package/dist/generated/prisma/models/WorksheetQuestionProgress.js +1 -0
  81. package/dist/generated/prisma/models/Workspace.d.ts +3620 -0
  82. package/dist/generated/prisma/models/Workspace.js +1 -0
  83. package/dist/generated/prisma/models/WorkspaceInvitation.d.ts +1490 -0
  84. package/dist/generated/prisma/models/WorkspaceInvitation.js +1 -0
  85. package/dist/generated/prisma/models/WorkspaceKnowledgeBase.d.ts +1410 -0
  86. package/dist/generated/prisma/models/WorkspaceKnowledgeBase.js +1 -0
  87. package/dist/generated/prisma/models/WorkspaceMember.d.ts +1326 -0
  88. package/dist/generated/prisma/models/WorkspaceMember.js +1 -0
  89. package/dist/generated/prisma/models.d.ts +39 -0
  90. package/dist/generated/prisma/models.js +1 -0
  91. package/dist/lib/ai/index.d.ts +3 -2
  92. package/dist/lib/ai/index.js +3 -2
  93. package/dist/lib/ai/llm-client.d.ts +1 -0
  94. package/dist/lib/ai/llm-client.js +17 -0
  95. package/dist/routers/_app.d.ts +40 -80
  96. package/dist/routers/auth.js +1 -1
  97. package/dist/routers/flashcards.d.ts +12 -1
  98. package/dist/routers/payment.d.ts +1 -12
  99. package/dist/routers/workspace.d.ts +27 -67
  100. package/dist/routers/workspace.js +1 -0
  101. package/dist/services/billing/payment.service.d.ts +1 -12
  102. package/dist/services/billing/payment.service.js +3 -6
  103. package/dist/services/billing/usage.service.d.ts +30 -10
  104. package/dist/services/billing/usage.service.js +87 -15
  105. package/dist/services/content/copilot.service.js +15 -29
  106. package/dist/services/content/flashcard-progress.service.js +9 -9
  107. package/dist/services/content/flashcard.service.d.ts +45 -1
  108. package/dist/services/content/flashcard.service.js +81 -68
  109. package/dist/services/content/media-analysis.service.js +27 -27
  110. package/dist/services/content/worksheet-generation.service.test.js +2 -2
  111. package/dist/services/workspace/workspace.service.d.ts +23 -67
  112. package/dist/services/workspace/workspace.service.js +69 -62
  113. package/dist/src/context.d.ts +27 -0
  114. package/dist/src/context.js +33 -0
  115. package/dist/src/index.d.ts +3 -0
  116. package/dist/src/index.js +1 -0
  117. package/dist/src/lib/ai/config.d.ts +20 -0
  118. package/dist/src/lib/ai/config.js +31 -0
  119. package/dist/src/lib/ai/embedding-client.d.ts +8 -0
  120. package/dist/src/lib/ai/embedding-client.js +30 -0
  121. package/dist/src/lib/ai/index.d.ts +48 -0
  122. package/dist/src/lib/ai/index.js +29 -0
  123. package/dist/src/lib/ai/inference-backend/client.d.ts +28 -0
  124. package/dist/src/lib/ai/inference-backend/client.js +301 -0
  125. package/dist/src/lib/ai/inference-backend/mocks.d.ts +12 -0
  126. package/dist/src/lib/ai/inference-backend/mocks.js +133 -0
  127. package/dist/src/lib/ai/inference-backend/types.d.ts +44 -0
  128. package/dist/src/lib/ai/inference-backend/types.js +1 -0
  129. package/dist/src/lib/ai/json-parse.d.ts +2 -0
  130. package/dist/src/lib/ai/json-parse.js +34 -0
  131. package/dist/src/lib/ai/llm-client.d.ts +7 -0
  132. package/dist/src/lib/ai/llm-client.js +36 -0
  133. package/dist/src/lib/ai/mock.d.ts +2 -0
  134. package/dist/src/lib/ai/mock.js +10 -0
  135. package/dist/src/lib/ai/types.d.ts +9 -0
  136. package/dist/src/lib/ai/types.js +1 -0
  137. package/dist/src/lib/auth.d.ts +36 -0
  138. package/dist/src/lib/auth.js +117 -0
  139. package/dist/src/lib/chunking.d.ts +19 -0
  140. package/dist/src/lib/chunking.js +47 -0
  141. package/dist/src/lib/constants.d.ts +13 -0
  142. package/dist/src/lib/constants.js +12 -0
  143. package/dist/src/lib/curated-kb-seed.d.ts +12 -0
  144. package/dist/src/lib/curated-kb-seed.js +155 -0
  145. package/dist/src/lib/email.d.ts +11 -0
  146. package/dist/src/lib/email.js +152 -0
  147. package/dist/src/lib/embeddings.d.ts +2 -0
  148. package/dist/src/lib/embeddings.js +1 -0
  149. package/dist/src/lib/ensure-curated-kb-catalog.d.ts +6 -0
  150. package/dist/src/lib/ensure-curated-kb-catalog.js +53 -0
  151. package/dist/src/lib/env.d.ts +41 -0
  152. package/dist/src/lib/env.js +57 -0
  153. package/dist/src/lib/errors.d.ts +33 -0
  154. package/dist/src/lib/errors.js +78 -0
  155. package/dist/src/lib/file.d.ts +0 -0
  156. package/dist/src/lib/file.js +1 -0
  157. package/dist/src/lib/inference.d.ts +1 -0
  158. package/dist/src/lib/inference.js +1 -0
  159. package/dist/src/lib/kb-meta.d.ts +8 -0
  160. package/dist/src/lib/kb-meta.js +77 -0
  161. package/dist/src/lib/logger.d.ts +62 -0
  162. package/dist/src/lib/logger.js +364 -0
  163. package/dist/src/lib/pdf.d.ts +11 -0
  164. package/dist/src/lib/pdf.js +11 -0
  165. package/dist/src/lib/prisma.d.ts +3 -0
  166. package/dist/src/lib/prisma.js +15 -0
  167. package/dist/src/lib/pusher.d.ts +38 -0
  168. package/dist/src/lib/pusher.js +170 -0
  169. package/dist/src/lib/retry.d.ts +15 -0
  170. package/dist/src/lib/retry.js +37 -0
  171. package/dist/src/lib/storage.d.ts +11 -0
  172. package/dist/src/lib/storage.js +71 -0
  173. package/dist/src/lib/stripe.d.ts +10 -0
  174. package/dist/src/lib/stripe.js +36 -0
  175. package/dist/src/lib/validation.d.ts +51 -0
  176. package/dist/src/lib/validation.js +64 -0
  177. package/dist/src/lib/workspace-kb.d.ts +5 -0
  178. package/dist/src/lib/workspace-kb.js +7 -0
  179. package/dist/src/repositories/artifact.repository.d.ts +64 -0
  180. package/dist/src/repositories/artifact.repository.js +40 -0
  181. package/dist/src/repositories/base.repository.d.ts +14 -0
  182. package/dist/src/repositories/base.repository.js +14 -0
  183. package/dist/src/repositories/invitation.repository.d.ts +104 -0
  184. package/dist/src/repositories/invitation.repository.js +44 -0
  185. package/dist/src/repositories/notification.repository.d.ts +76 -0
  186. package/dist/src/repositories/notification.repository.js +44 -0
  187. package/dist/src/repositories/user.repository.d.ts +84 -0
  188. package/dist/src/repositories/user.repository.js +37 -0
  189. package/dist/src/repositories/workspace-member.repository.d.ts +35 -0
  190. package/dist/src/repositories/workspace-member.repository.js +31 -0
  191. package/dist/src/repositories/workspace.repository.d.ts +101 -0
  192. package/dist/src/repositories/workspace.repository.js +79 -0
  193. package/dist/src/routers/_app.d.ts +3464 -0
  194. package/dist/src/routers/_app.js +36 -0
  195. package/dist/src/routers/admin.d.ts +358 -0
  196. package/dist/src/routers/admin.js +105 -0
  197. package/dist/src/routers/annotations.d.ts +219 -0
  198. package/dist/src/routers/annotations.js +29 -0
  199. package/dist/src/routers/artifactVersions.d.ts +65 -0
  200. package/dist/src/routers/artifactVersions.js +14 -0
  201. package/dist/src/routers/auth.d.ts +161 -0
  202. package/dist/src/routers/auth.js +97 -0
  203. package/dist/src/routers/chat.d.ts +170 -0
  204. package/dist/src/routers/chat.js +32 -0
  205. package/dist/src/routers/copilot.d.ts +200 -0
  206. package/dist/src/routers/copilot.js +52 -0
  207. package/dist/src/routers/flashcards.d.ts +336 -0
  208. package/dist/src/routers/flashcards.js +93 -0
  209. package/dist/src/routers/knowledgeBase.d.ts +421 -0
  210. package/dist/src/routers/knowledgeBase.js +118 -0
  211. package/dist/src/routers/members.d.ts +169 -0
  212. package/dist/src/routers/members.js +47 -0
  213. package/dist/src/routers/notifications.d.ts +99 -0
  214. package/dist/src/routers/notifications.js +25 -0
  215. package/dist/src/routers/payment.d.ts +80 -0
  216. package/dist/src/routers/payment.js +21 -0
  217. package/dist/src/routers/podcast.d.ts +287 -0
  218. package/dist/src/routers/podcast.js +34 -0
  219. package/dist/src/routers/studyguide.d.ts +36 -0
  220. package/dist/src/routers/studyguide.js +8 -0
  221. package/dist/src/routers/worksheets.d.ts +429 -0
  222. package/dist/src/routers/worksheets.js +139 -0
  223. package/dist/src/routers/workspace.d.ts +563 -0
  224. package/dist/src/routers/workspace.js +104 -0
  225. package/dist/src/scripts/purge-deleted-users.d.ts +1 -0
  226. package/dist/src/scripts/purge-deleted-users.js +148 -0
  227. package/dist/src/server.d.ts +1 -0
  228. package/dist/src/server.js +190 -0
  229. package/dist/src/services/activity/activity-human-description.service.d.ts +13 -0
  230. package/dist/src/services/activity/activity-human-description.service.js +221 -0
  231. package/dist/src/services/activity/activity-human-description.service.test.d.ts +1 -0
  232. package/dist/src/services/activity/activity-human-description.service.test.js +16 -0
  233. package/dist/src/services/activity/activity-log.service.d.ts +87 -0
  234. package/dist/src/services/activity/activity-log.service.js +276 -0
  235. package/dist/src/services/activity/activity-log.service.test.d.ts +1 -0
  236. package/dist/src/services/activity/activity-log.service.test.js +27 -0
  237. package/dist/src/services/admin/admin.service.d.ts +270 -0
  238. package/dist/src/services/admin/admin.service.js +476 -0
  239. package/dist/src/services/ai/ai-session.service.d.ts +5 -0
  240. package/dist/src/services/ai/ai-session.service.js +4 -0
  241. package/dist/src/services/artifacts/annotation.service.d.ts +177 -0
  242. package/dist/src/services/artifacts/annotation.service.js +154 -0
  243. package/dist/src/services/artifacts/artifact-version.service.d.ts +38 -0
  244. package/dist/src/services/artifacts/artifact-version.service.js +129 -0
  245. package/dist/src/services/artifacts/chat.service.d.ts +127 -0
  246. package/dist/src/services/artifacts/chat.service.js +182 -0
  247. package/dist/src/services/artifacts/study-guide.service.d.ts +18 -0
  248. package/dist/src/services/artifacts/study-guide.service.js +65 -0
  249. package/dist/src/services/auth/auth.service.d.ts +94 -0
  250. package/dist/src/services/auth/auth.service.js +368 -0
  251. package/dist/src/services/base.service.d.ts +14 -0
  252. package/dist/src/services/base.service.js +14 -0
  253. package/dist/src/services/billing/payment.service.d.ts +44 -0
  254. package/dist/src/services/billing/payment.service.js +365 -0
  255. package/dist/src/services/billing/subscription.service.d.ts +37 -0
  256. package/dist/src/services/billing/subscription.service.js +654 -0
  257. package/dist/src/services/billing/usage.service.d.ts +47 -0
  258. package/dist/src/services/billing/usage.service.js +149 -0
  259. package/dist/src/services/content/copilot.service.d.ts +113 -0
  260. package/dist/src/services/content/copilot.service.js +439 -0
  261. package/dist/src/services/content/flashcard-progress.service.d.ts +159 -0
  262. package/dist/src/services/content/flashcard-progress.service.js +432 -0
  263. package/dist/src/services/content/flashcard.service.d.ts +184 -0
  264. package/dist/src/services/content/flashcard.service.js +339 -0
  265. package/dist/src/services/content/media-analysis.service.d.ts +23 -0
  266. package/dist/src/services/content/media-analysis.service.js +404 -0
  267. package/dist/src/services/content/podcast.service.d.ts +267 -0
  268. package/dist/src/services/content/podcast.service.js +653 -0
  269. package/dist/src/services/content/worksheet-content.service.d.ts +37 -0
  270. package/dist/src/services/content/worksheet-content.service.js +84 -0
  271. package/dist/src/services/content/worksheet-content.service.test.d.ts +1 -0
  272. package/dist/src/services/content/worksheet-content.service.test.js +69 -0
  273. package/dist/src/services/content/worksheet-generation.service.d.ts +91 -0
  274. package/dist/src/services/content/worksheet-generation.service.js +95 -0
  275. package/dist/src/services/content/worksheet-generation.service.test.d.ts +1 -0
  276. package/dist/src/services/content/worksheet-generation.service.test.js +20 -0
  277. package/dist/src/services/content/worksheet.service.d.ts +347 -0
  278. package/dist/src/services/content/worksheet.service.js +599 -0
  279. package/dist/src/services/knowledge/knowledge-base.service.d.ts +316 -0
  280. package/dist/src/services/knowledge/knowledge-base.service.js +544 -0
  281. package/dist/src/services/members/invitation.service.d.ts +66 -0
  282. package/dist/src/services/members/invitation.service.js +348 -0
  283. package/dist/src/services/members/member.service.d.ts +36 -0
  284. package/dist/src/services/members/member.service.js +193 -0
  285. package/dist/src/services/notifications/notification.service.d.ts +214 -0
  286. package/dist/src/services/notifications/notification.service.js +550 -0
  287. package/dist/src/services/notifications/notification.service.test.d.ts +1 -0
  288. package/dist/src/services/notifications/notification.service.test.js +87 -0
  289. package/dist/src/services/workspace/workspace-analytics.service.d.ts +24 -0
  290. package/dist/src/services/workspace/workspace-analytics.service.js +95 -0
  291. package/dist/src/services/workspace/workspace-kb.service.d.ts +40 -0
  292. package/dist/src/services/workspace/workspace-kb.service.js +184 -0
  293. package/dist/src/services/workspace/workspace.service.d.ts +263 -0
  294. package/dist/src/services/workspace/workspace.service.js +401 -0
  295. package/dist/src/trpc.d.ts +60 -0
  296. package/dist/src/trpc.js +217 -0
  297. package/dist/src/types/index.d.ts +126 -0
  298. package/dist/src/types/index.js +1 -0
  299. package/dist/trpc.d.ts +12 -4
  300. package/dist/trpc.js +5 -11
  301. package/package.json +8 -9
  302. package/prisma/schema.prisma +3 -4
  303. package/prisma/seed.mjs +5 -2
  304. package/prisma.config.ts +16 -0
  305. package/src/context.ts +33 -3
  306. package/src/lib/ai/index.ts +3 -0
  307. package/src/lib/ai/llm-client.ts +23 -0
  308. package/src/lib/prisma.ts +18 -9
  309. package/src/routers/auth.ts +1 -1
  310. package/src/routers/workspace.ts +4 -0
  311. package/src/scripts/purge-deleted-users.ts +1 -3
  312. package/src/services/billing/payment.service.ts +3 -6
  313. package/src/services/billing/usage.service.ts +190 -77
  314. package/src/services/content/copilot.service.ts +23 -32
  315. package/src/services/content/flashcard-progress.service.ts +12 -9
  316. package/src/services/content/flashcard.service.ts +89 -66
  317. package/src/services/content/media-analysis.service.ts +34 -29
  318. package/src/services/content/worksheet-generation.service.test.ts +2 -2
  319. package/src/services/workspace/workspace.service.ts +73 -66
  320. package/src/trpc.ts +5 -13
  321. package/tsconfig.json +3 -0
@@ -0,0 +1,476 @@
1
+ import { BaseService } from '../base.service.js';
2
+ import { stripe } from '../../lib/stripe.js';
3
+ import { buildActivityLogWhere, deleteActivityLogsOlderThan, getActivityRetentionDays, } from '../activity/activity-log.service.js';
4
+ import { getActivityHumanDescription } from '../activity/activity-human-description.service.js';
5
+ function csvEscapeCell(value) {
6
+ if (value === null || value === undefined)
7
+ return '';
8
+ const s = String(value);
9
+ if (/[",\n\r]/.test(s))
10
+ return `"${s.replace(/"/g, '""')}"`;
11
+ return s;
12
+ }
13
+ export class AdminService extends BaseService {
14
+ constructor(db) {
15
+ super(db);
16
+ }
17
+ async getSystemStats() {
18
+ const totalUsers = await this.db.user.count();
19
+ const totalWorkspaces = await this.db.workspace.count();
20
+ const totalSubscriptions = await this.db.subscription.count({
21
+ where: { status: 'active' },
22
+ });
23
+ const subRevenueResult = await this.db.invoice.aggregate({
24
+ where: { status: 'paid', type: 'SUBSCRIPTION' },
25
+ _sum: { amountPaid: true },
26
+ });
27
+ const topupRevenueResult = await this.db.invoice.aggregate({
28
+ where: { status: 'paid', type: 'TOPUP' },
29
+ _sum: { amountPaid: true },
30
+ });
31
+ const subRevenue = Number(subRevenueResult._sum.amountPaid || 0) / 100;
32
+ const topupRevenue = Number(topupRevenueResult._sum.amountPaid || 0) / 100;
33
+ return {
34
+ totalUsers,
35
+ totalWorkspaces,
36
+ totalSubscriptions,
37
+ revenue: subRevenue + topupRevenue,
38
+ subscriptionRevenue: subRevenue,
39
+ topupRevenue,
40
+ };
41
+ }
42
+ async listUsers(input) {
43
+ const search = input.search?.trim();
44
+ const baseRoleWhere = {
45
+ role: { name: { not: 'System Admin' } },
46
+ };
47
+ const searchWhere = search && search.length > 0
48
+ ? {
49
+ OR: [
50
+ { email: { contains: search, mode: 'insensitive' } },
51
+ { name: { contains: search, mode: 'insensitive' } },
52
+ { id: { contains: search } },
53
+ ],
54
+ }
55
+ : undefined;
56
+ const verifiedWhere = input.emailVerified === 'yes'
57
+ ? { emailVerified: { not: null } }
58
+ : input.emailVerified === 'no'
59
+ ? { emailVerified: null }
60
+ : undefined;
61
+ const joinedWhere = input.joinedFrom || input.joinedTo
62
+ ? {
63
+ createdAt: {
64
+ ...(input.joinedFrom ? { gte: input.joinedFrom } : {}),
65
+ ...(input.joinedTo ? { lte: input.joinedTo } : {}),
66
+ },
67
+ }
68
+ : undefined;
69
+ const where = {
70
+ AND: [
71
+ baseRoleWhere,
72
+ ...(searchWhere ? [searchWhere] : []),
73
+ ...(verifiedWhere ? [verifiedWhere] : []),
74
+ ...(joinedWhere ? [joinedWhere] : []),
75
+ ],
76
+ };
77
+ const totalCount = await this.db.user.count({ where });
78
+ const pageSize = input.pageSize;
79
+ const totalPages = Math.max(1, Math.ceil(totalCount / pageSize));
80
+ const page = Math.min(Math.max(1, input.page), totalPages);
81
+ const skip = (page - 1) * pageSize;
82
+ const users = await this.db.user.findMany({
83
+ where,
84
+ skip,
85
+ take: pageSize,
86
+ orderBy: { createdAt: 'desc' },
87
+ include: {
88
+ role: true,
89
+ profilePicture: true,
90
+ subscriptions: {
91
+ orderBy: { createdAt: 'desc' },
92
+ take: 1,
93
+ include: { plan: true },
94
+ },
95
+ },
96
+ });
97
+ return {
98
+ users: users.map((user) => ({
99
+ ...user,
100
+ profilePicture: user.profilePicture?.objectKey
101
+ ? `/profile-picture/${user.profilePicture.objectKey}?t=${new Date(user.updatedAt).getTime()}`
102
+ : null,
103
+ })),
104
+ page,
105
+ pageSize,
106
+ totalCount,
107
+ totalPages,
108
+ };
109
+ }
110
+ async listInvoices(input) {
111
+ const search = input.search?.trim();
112
+ const searchWhere = search && search.length > 0
113
+ ? {
114
+ OR: [
115
+ { id: { contains: search } },
116
+ { stripeInvoiceId: { contains: search, mode: 'insensitive' } },
117
+ { user: { email: { contains: search, mode: 'insensitive' } } },
118
+ { user: { name: { contains: search, mode: 'insensitive' } } },
119
+ ],
120
+ }
121
+ : undefined;
122
+ const userIdWhere = input.userId?.trim() ? { userId: input.userId.trim() } : undefined;
123
+ const statusWhere = input.status?.trim() ? { status: input.status.trim() } : undefined;
124
+ const typeWhere = input.type ? { type: input.type } : undefined;
125
+ const dateWhere = input.from || input.to
126
+ ? {
127
+ createdAt: {
128
+ ...(input.from ? { gte: input.from } : {}),
129
+ ...(input.to ? { lte: input.to } : {}),
130
+ },
131
+ }
132
+ : undefined;
133
+ const andParts = [
134
+ ...(searchWhere ? [searchWhere] : []),
135
+ ...(userIdWhere ? [userIdWhere] : []),
136
+ ...(statusWhere ? [statusWhere] : []),
137
+ ...(typeWhere ? [typeWhere] : []),
138
+ ...(dateWhere ? [dateWhere] : []),
139
+ ];
140
+ const where = andParts.length > 0 ? { AND: andParts } : {};
141
+ const totalCount = await this.db.invoice.count({ where });
142
+ const pageSize = input.pageSize;
143
+ const totalPages = Math.max(1, Math.ceil(totalCount / pageSize));
144
+ const page = Math.min(Math.max(1, input.page), totalPages);
145
+ const skip = (page - 1) * pageSize;
146
+ const invoices = await this.db.invoice.findMany({
147
+ where,
148
+ skip,
149
+ take: pageSize,
150
+ orderBy: { createdAt: 'desc' },
151
+ include: {
152
+ user: { include: { profilePicture: true } },
153
+ subscription: { include: { plan: true } },
154
+ },
155
+ });
156
+ const items = invoices.map((invoice) => ({
157
+ ...invoice,
158
+ user: invoice.user
159
+ ? {
160
+ ...invoice.user,
161
+ profilePicture: invoice.user.profilePicture?.objectKey
162
+ ? `/profile-picture/${invoice.user.profilePicture.objectKey}?t=${new Date(invoice.user.updatedAt).getTime()}`
163
+ : null,
164
+ }
165
+ : invoice.user,
166
+ }));
167
+ return { items, page, pageSize, totalCount, totalPages };
168
+ }
169
+ async listWorkspaces(input) {
170
+ const workspaces = await this.db.workspace.findMany({
171
+ take: input.limit + 1,
172
+ cursor: input.cursor ? { id: input.cursor } : undefined,
173
+ orderBy: { createdAt: 'desc' },
174
+ include: {
175
+ owner: { select: { name: true, email: true } },
176
+ },
177
+ });
178
+ let nextCursor = undefined;
179
+ if (workspaces.length > input.limit) {
180
+ const nextItem = workspaces.pop();
181
+ nextCursor = nextItem.id;
182
+ }
183
+ return { workspaces, nextCursor };
184
+ }
185
+ async updateUserRole(actorId, input) {
186
+ const role = await this.db.role.findUnique({ where: { name: input.roleName } });
187
+ if (!role) {
188
+ throw new Error(`Role ${input.roleName} not found`);
189
+ }
190
+ const updatedUser = await this.db.user.update({
191
+ where: { id: input.userId },
192
+ data: { roleId: role.id },
193
+ });
194
+ this.logger.info(`User ${input.userId} role updated to ${input.roleName} by admin ${actorId}`, 'ADMIN');
195
+ return updatedUser;
196
+ }
197
+ listPlans() {
198
+ return this.db.plan.findMany({
199
+ include: { limit: true },
200
+ orderBy: { price: 'asc' },
201
+ });
202
+ }
203
+ async upsertPlan(actorId, input) {
204
+ let { limits, id, stripePriceId, ...planData } = input;
205
+ if (!stripePriceId && !id) {
206
+ if (!stripe) {
207
+ throw new Error('Stripe is not configured on the server');
208
+ }
209
+ try {
210
+ const product = await stripe.products.create({
211
+ name: planData.name,
212
+ description: planData.description,
213
+ });
214
+ const price = await stripe.prices.create({
215
+ product: product.id,
216
+ unit_amount: Math.round(planData.price * 100),
217
+ currency: 'usd',
218
+ recurring: planData.type === 'subscription'
219
+ ? { interval: planData.interval || 'month' }
220
+ : undefined,
221
+ });
222
+ stripePriceId = price.id;
223
+ this.logger.info(`Automatically created Stripe Product (${product.id}) and Price (${price.id}) for plan: ${planData.name}`, 'STRIPE');
224
+ }
225
+ catch (err) {
226
+ this.logger.error(`Failed to automate Stripe creation: ${err.message}`, 'STRIPE');
227
+ throw new Error(`Stripe error: ${err.message}`);
228
+ }
229
+ }
230
+ if (id) {
231
+ const updatedPlan = await this.db.plan.update({
232
+ where: { id },
233
+ data: {
234
+ ...planData,
235
+ stripePriceId,
236
+ limit: {
237
+ update: {
238
+ maxStorageBytes: BigInt(limits.maxStorageBytes),
239
+ maxWorksheets: limits.maxWorksheets,
240
+ maxFlashcards: limits.maxFlashcards,
241
+ maxPodcasts: limits.maxPodcasts,
242
+ maxStudyGuides: limits.maxStudyGuides,
243
+ },
244
+ },
245
+ },
246
+ });
247
+ this.logger.info(`Plan ${id} updated by admin ${actorId}`, 'ADMIN');
248
+ return updatedPlan;
249
+ }
250
+ const newPlan = await this.db.plan.create({
251
+ data: {
252
+ ...planData,
253
+ stripePriceId: stripePriceId || '',
254
+ limit: {
255
+ create: {
256
+ maxStorageBytes: BigInt(limits.maxStorageBytes),
257
+ maxWorksheets: limits.maxWorksheets,
258
+ maxFlashcards: limits.maxFlashcards,
259
+ maxPodcasts: limits.maxPodcasts,
260
+ maxStudyGuides: limits.maxStudyGuides,
261
+ },
262
+ },
263
+ },
264
+ });
265
+ this.logger.info(`New plan ${newPlan.id} created by admin ${actorId}`, 'ADMIN');
266
+ return newPlan;
267
+ }
268
+ async deletePlan(actorId, id) {
269
+ const activeSubs = await this.db.subscription.count({
270
+ where: { planId: id, status: 'active' },
271
+ });
272
+ if (activeSubs > 0) {
273
+ await this.db.plan.update({ where: { id }, data: { active: false } });
274
+ return {
275
+ success: true,
276
+ message: 'Plan deactivated because it has active subscribers.',
277
+ };
278
+ }
279
+ await this.db.plan.delete({ where: { id } });
280
+ this.logger.info(`Plan ${id} deleted by admin ${actorId}`, 'ADMIN');
281
+ return { success: true, message: 'Plan deleted.' };
282
+ }
283
+ getUserInvoices(input) {
284
+ return this.db.invoice.findMany({
285
+ where: { userId: input.userId },
286
+ orderBy: { createdAt: 'desc' },
287
+ take: input.limit,
288
+ include: {
289
+ subscription: { include: { plan: true } },
290
+ },
291
+ });
292
+ }
293
+ async getUserDetailedInfo(userId) {
294
+ const user = await this.db.user.findUnique({
295
+ where: { id: userId },
296
+ include: {
297
+ role: true,
298
+ profilePicture: true,
299
+ subscriptions: {
300
+ orderBy: { createdAt: 'desc' },
301
+ take: 1,
302
+ include: { plan: true },
303
+ },
304
+ _count: {
305
+ select: { workspaces: true, artifacts: true },
306
+ },
307
+ },
308
+ });
309
+ if (!user)
310
+ throw new Error('User not found');
311
+ const totalSpentResult = await this.db.invoice.aggregate({
312
+ where: { userId, status: 'paid' },
313
+ _sum: { amountPaid: true },
314
+ });
315
+ const totalSpent = Number(totalSpentResult._sum.amountPaid || 0) / 100;
316
+ const purchasedCredits = await this.db.userCredit.groupBy({
317
+ by: ['resourceType'],
318
+ where: { userId },
319
+ _sum: { amount: true },
320
+ });
321
+ const profilePictureUrl = user.profilePicture?.objectKey
322
+ ? `/profile-picture/${user.profilePicture.objectKey}?t=${new Date(user.updatedAt).getTime()}`
323
+ : null;
324
+ return {
325
+ ...user,
326
+ totalSpent,
327
+ purchasedCredits: purchasedCredits.map((c) => ({
328
+ type: c.resourceType,
329
+ amount: c._sum.amount || 0,
330
+ })),
331
+ profilePicture: profilePictureUrl,
332
+ };
333
+ }
334
+ async debugInvoices() {
335
+ const count = await this.db.invoice.count();
336
+ const all = await this.db.invoice.findMany({ take: 10 });
337
+ return { count, all };
338
+ }
339
+ listResourcePrices() {
340
+ return this.db.resourcePrice.findMany({
341
+ orderBy: { resourceType: 'asc' },
342
+ });
343
+ }
344
+ async upsertResourcePrice(actorId, input) {
345
+ const price = await this.db.resourcePrice.upsert({
346
+ where: { resourceType: input.resourceType },
347
+ update: { priceCents: input.priceCents },
348
+ create: {
349
+ resourceType: input.resourceType,
350
+ priceCents: input.priceCents,
351
+ },
352
+ });
353
+ this.logger.info(`Resource price for ${input.resourceType} updated to ${input.priceCents} by admin ${actorId}`, 'ADMIN');
354
+ return price;
355
+ }
356
+ async listRecentInvoices(limit) {
357
+ const invoices = await this.db.invoice.findMany({
358
+ take: limit,
359
+ orderBy: { createdAt: 'desc' },
360
+ include: {
361
+ user: { include: { profilePicture: true } },
362
+ },
363
+ });
364
+ return invoices.map((invoice) => ({
365
+ ...invoice,
366
+ user: {
367
+ ...invoice.user,
368
+ profilePicture: invoice.user.profilePicture?.objectKey
369
+ ? `/profile-picture/${invoice.user.profilePicture.objectKey}?t=${new Date(invoice.user.updatedAt).getTime()}`
370
+ : null,
371
+ },
372
+ }));
373
+ }
374
+ async activityList(input) {
375
+ const where = buildActivityLogWhere({
376
+ from: input.from,
377
+ to: input.to,
378
+ actorUserId: input.actorUserId,
379
+ workspaceId: input.workspaceId,
380
+ category: input.category,
381
+ status: input.status,
382
+ search: input.search,
383
+ });
384
+ const totalCount = await this.db.activityLog.count({ where });
385
+ const totalPages = Math.max(1, Math.ceil(totalCount / input.limit));
386
+ const page = Math.min(Math.max(1, input.page), totalPages);
387
+ const skip = (page - 1) * input.limit;
388
+ const rows = await this.db.activityLog.findMany({
389
+ where,
390
+ skip,
391
+ take: input.limit,
392
+ orderBy: { createdAt: 'desc' },
393
+ include: {
394
+ actor: { select: { id: true, email: true, name: true } },
395
+ workspace: { select: { id: true, title: true } },
396
+ },
397
+ });
398
+ return {
399
+ items: rows.map((r) => ({
400
+ id: r.id,
401
+ createdAt: r.createdAt,
402
+ action: r.action,
403
+ description: getActivityHumanDescription(r.trpcPath, r.action),
404
+ category: r.category,
405
+ trpcPath: r.trpcPath,
406
+ status: r.status,
407
+ durationMs: r.durationMs,
408
+ errorCode: r.errorCode,
409
+ ipAddress: r.ipAddress,
410
+ actor: r.actor,
411
+ workspace: r.workspace,
412
+ metadata: r.metadata,
413
+ })),
414
+ page,
415
+ pageSize: input.limit,
416
+ totalCount,
417
+ totalPages,
418
+ };
419
+ }
420
+ async activityExportCsv(input) {
421
+ const where = buildActivityLogWhere({
422
+ from: input.from,
423
+ to: input.to,
424
+ actorUserId: input.actorUserId,
425
+ workspaceId: input.workspaceId,
426
+ category: input.category,
427
+ status: input.status,
428
+ search: input.search,
429
+ });
430
+ const rows = await this.db.activityLog.findMany({
431
+ where,
432
+ orderBy: { createdAt: 'desc' },
433
+ take: input.maxRows,
434
+ include: {
435
+ actor: { select: { email: true, name: true } },
436
+ workspace: { select: { title: true } },
437
+ },
438
+ });
439
+ const header = [
440
+ 'createdAt',
441
+ 'actorEmail',
442
+ 'actorName',
443
+ 'description',
444
+ 'trpcPath',
445
+ 'category',
446
+ 'status',
447
+ 'durationMs',
448
+ 'workspaceTitle',
449
+ 'errorCode',
450
+ ].join(',');
451
+ const lines = rows.map((r) => [
452
+ csvEscapeCell(r.createdAt.toISOString()),
453
+ csvEscapeCell(r.actor?.email),
454
+ csvEscapeCell(r.actor?.name),
455
+ csvEscapeCell(getActivityHumanDescription(r.trpcPath, r.action)),
456
+ csvEscapeCell(r.trpcPath),
457
+ csvEscapeCell(r.category),
458
+ csvEscapeCell(r.status),
459
+ csvEscapeCell(r.durationMs),
460
+ csvEscapeCell(r.workspace?.title),
461
+ csvEscapeCell(r.errorCode),
462
+ ].join(','));
463
+ return {
464
+ csv: [header, ...lines].join('\n'),
465
+ count: rows.length,
466
+ };
467
+ }
468
+ async activityPurgeRetention() {
469
+ const days = getActivityRetentionDays();
470
+ const cutoff = new Date();
471
+ cutoff.setDate(cutoff.getDate() - days);
472
+ const { deleted } = await deleteActivityLogsOlderThan(this.db, cutoff);
473
+ this.logger.info(`ActivityLog retention purge: deleted ${deleted} rows older than ${days} days (cutoff ${cutoff.toISOString()})`, 'ADMIN');
474
+ return { deleted, retentionDays: days, cutoff };
475
+ }
476
+ }
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Back-compat shim — prefer `import { ai } from '../../lib/ai/index.js'`.
3
+ */
4
+ export { inferenceBackend as aiSessionService } from '../../lib/ai/inference-backend/client.js';
5
+ export type { AISession, ProcessFileResult } from '../../lib/ai/inference-backend/types.js';
@@ -0,0 +1,4 @@
1
+ /**
2
+ * Back-compat shim — prefer `import { ai } from '../../lib/ai/index.js'`.
3
+ */
4
+ export { inferenceBackend as aiSessionService } from '../../lib/ai/inference-backend/client.js';
@@ -0,0 +1,177 @@
1
+ import type { PrismaClient } from '@prisma/client';
2
+ import { BaseService } from '../base.service.js';
3
+ export declare class AnnotationService extends BaseService {
4
+ constructor(db: PrismaClient);
5
+ listHighlights(artifactVersionId: string): import("../../../generated/prisma/internal/prismaNamespace.js").PrismaPromise<({
6
+ comments: ({
7
+ user: {
8
+ id: string;
9
+ name: string | null;
10
+ profilePicture: {
11
+ userId: string | null;
12
+ id: string;
13
+ name: string;
14
+ createdAt: Date;
15
+ size: number;
16
+ workspaceId: string | null;
17
+ url: string | null;
18
+ objectKey: string | null;
19
+ mimeType: string;
20
+ bucket: string | null;
21
+ checksum: string | null;
22
+ meta: import("@prisma/client/runtime/client").JsonValue | null;
23
+ aiTranscription: import("@prisma/client/runtime/client").JsonValue | null;
24
+ } | null;
25
+ };
26
+ } & {
27
+ userId: string;
28
+ id: string;
29
+ createdAt: Date;
30
+ updatedAt: Date;
31
+ highlightId: string;
32
+ content: string;
33
+ })[];
34
+ user: {
35
+ id: string;
36
+ name: string | null;
37
+ profilePicture: {
38
+ userId: string | null;
39
+ id: string;
40
+ name: string;
41
+ createdAt: Date;
42
+ size: number;
43
+ workspaceId: string | null;
44
+ url: string | null;
45
+ objectKey: string | null;
46
+ mimeType: string;
47
+ bucket: string | null;
48
+ checksum: string | null;
49
+ meta: import("@prisma/client/runtime/client").JsonValue | null;
50
+ aiTranscription: import("@prisma/client/runtime/client").JsonValue | null;
51
+ } | null;
52
+ };
53
+ } & {
54
+ userId: string;
55
+ id: string;
56
+ createdAt: Date;
57
+ updatedAt: Date;
58
+ color: string;
59
+ artifactVersionId: string;
60
+ startOffset: number;
61
+ endOffset: number;
62
+ selectedText: string;
63
+ })[]>;
64
+ createHighlight(userId: string, input: {
65
+ artifactVersionId: string;
66
+ startOffset: number;
67
+ endOffset: number;
68
+ selectedText: string;
69
+ color?: string;
70
+ }): Promise<{
71
+ comments: {
72
+ userId: string;
73
+ id: string;
74
+ createdAt: Date;
75
+ updatedAt: Date;
76
+ highlightId: string;
77
+ content: string;
78
+ }[];
79
+ user: {
80
+ id: string;
81
+ name: string | null;
82
+ profilePicture: {
83
+ userId: string | null;
84
+ id: string;
85
+ name: string;
86
+ createdAt: Date;
87
+ size: number;
88
+ workspaceId: string | null;
89
+ url: string | null;
90
+ objectKey: string | null;
91
+ mimeType: string;
92
+ bucket: string | null;
93
+ checksum: string | null;
94
+ meta: import("@prisma/client/runtime/client").JsonValue | null;
95
+ aiTranscription: import("@prisma/client/runtime/client").JsonValue | null;
96
+ } | null;
97
+ };
98
+ } & {
99
+ userId: string;
100
+ id: string;
101
+ createdAt: Date;
102
+ updatedAt: Date;
103
+ color: string;
104
+ artifactVersionId: string;
105
+ startOffset: number;
106
+ endOffset: number;
107
+ selectedText: string;
108
+ }>;
109
+ deleteHighlight(userId: string, highlightId: string): Promise<{
110
+ success: boolean;
111
+ }>;
112
+ addComment(userId: string, input: {
113
+ highlightId: string;
114
+ content: string;
115
+ }): Promise<{
116
+ user: {
117
+ id: string;
118
+ name: string | null;
119
+ profilePicture: {
120
+ userId: string | null;
121
+ id: string;
122
+ name: string;
123
+ createdAt: Date;
124
+ size: number;
125
+ workspaceId: string | null;
126
+ url: string | null;
127
+ objectKey: string | null;
128
+ mimeType: string;
129
+ bucket: string | null;
130
+ checksum: string | null;
131
+ meta: import("@prisma/client/runtime/client").JsonValue | null;
132
+ aiTranscription: import("@prisma/client/runtime/client").JsonValue | null;
133
+ } | null;
134
+ };
135
+ } & {
136
+ userId: string;
137
+ id: string;
138
+ createdAt: Date;
139
+ updatedAt: Date;
140
+ highlightId: string;
141
+ content: string;
142
+ }>;
143
+ updateComment(userId: string, input: {
144
+ commentId: string;
145
+ content: string;
146
+ }): Promise<{
147
+ user: {
148
+ id: string;
149
+ name: string | null;
150
+ profilePicture: {
151
+ userId: string | null;
152
+ id: string;
153
+ name: string;
154
+ createdAt: Date;
155
+ size: number;
156
+ workspaceId: string | null;
157
+ url: string | null;
158
+ objectKey: string | null;
159
+ mimeType: string;
160
+ bucket: string | null;
161
+ checksum: string | null;
162
+ meta: import("@prisma/client/runtime/client").JsonValue | null;
163
+ aiTranscription: import("@prisma/client/runtime/client").JsonValue | null;
164
+ } | null;
165
+ };
166
+ } & {
167
+ userId: string;
168
+ id: string;
169
+ createdAt: Date;
170
+ updatedAt: Date;
171
+ highlightId: string;
172
+ content: string;
173
+ }>;
174
+ deleteComment(userId: string, commentId: string): Promise<{
175
+ success: boolean;
176
+ }>;
177
+ }