@goscribe/server 1.6.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 (281) hide show
  1. package/dist/generated/prisma/client.d.ts +224 -0
  2. package/dist/generated/prisma/client.js +34 -0
  3. package/dist/generated/prisma/commonInputTypes.d.ts +941 -0
  4. package/dist/generated/prisma/commonInputTypes.js +10 -0
  5. package/dist/generated/prisma/enums.d.ts +67 -0
  6. package/dist/generated/prisma/enums.js +66 -0
  7. package/dist/generated/prisma/internal/class.d.ts +539 -0
  8. package/dist/generated/prisma/internal/class.js +49 -0
  9. package/dist/generated/prisma/internal/prismaNamespace.d.ts +3924 -0
  10. package/dist/generated/prisma/internal/prismaNamespace.js +557 -0
  11. package/dist/generated/prisma/models/ActivityLog.d.ts +1847 -0
  12. package/dist/generated/prisma/models/ActivityLog.js +1 -0
  13. package/dist/generated/prisma/models/Artifact.d.ts +2345 -0
  14. package/dist/generated/prisma/models/Artifact.js +1 -0
  15. package/dist/generated/prisma/models/ArtifactVersion.d.ts +1550 -0
  16. package/dist/generated/prisma/models/ArtifactVersion.js +1 -0
  17. package/dist/generated/prisma/models/Channel.d.ts +1257 -0
  18. package/dist/generated/prisma/models/Channel.js +1 -0
  19. package/dist/generated/prisma/models/Chat.d.ts +1339 -0
  20. package/dist/generated/prisma/models/Chat.js +1 -0
  21. package/dist/generated/prisma/models/CopilotConversation.d.ts +1450 -0
  22. package/dist/generated/prisma/models/CopilotConversation.js +1 -0
  23. package/dist/generated/prisma/models/CopilotMessage.d.ts +1179 -0
  24. package/dist/generated/prisma/models/CopilotMessage.js +1 -0
  25. package/dist/generated/prisma/models/FileAsset.d.ts +1832 -0
  26. package/dist/generated/prisma/models/FileAsset.js +1 -0
  27. package/dist/generated/prisma/models/Flashcard.d.ts +1460 -0
  28. package/dist/generated/prisma/models/Flashcard.js +1 -0
  29. package/dist/generated/prisma/models/FlashcardProgress.d.ts +1782 -0
  30. package/dist/generated/prisma/models/FlashcardProgress.js +1 -0
  31. package/dist/generated/prisma/models/Folder.d.ts +1685 -0
  32. package/dist/generated/prisma/models/Folder.js +1 -0
  33. package/dist/generated/prisma/models/IdempotencyRecord.d.ts +1319 -0
  34. package/dist/generated/prisma/models/IdempotencyRecord.js +1 -0
  35. package/dist/generated/prisma/models/Invoice.d.ts +1586 -0
  36. package/dist/generated/prisma/models/Invoice.js +1 -0
  37. package/dist/generated/prisma/models/KnowledgeBase.d.ts +1721 -0
  38. package/dist/generated/prisma/models/KnowledgeBase.js +1 -0
  39. package/dist/generated/prisma/models/KnowledgeBaseChunk.d.ts +1333 -0
  40. package/dist/generated/prisma/models/KnowledgeBaseChunk.js +1 -0
  41. package/dist/generated/prisma/models/KnowledgeBaseDocument.d.ts +1695 -0
  42. package/dist/generated/prisma/models/KnowledgeBaseDocument.js +1 -0
  43. package/dist/generated/prisma/models/Notification.d.ts +1992 -0
  44. package/dist/generated/prisma/models/Notification.js +1 -0
  45. package/dist/generated/prisma/models/PasswordResetToken.d.ts +1210 -0
  46. package/dist/generated/prisma/models/PasswordResetToken.js +1 -0
  47. package/dist/generated/prisma/models/Plan.d.ts +1431 -0
  48. package/dist/generated/prisma/models/Plan.js +1 -0
  49. package/dist/generated/prisma/models/PlanLimit.d.ts +1328 -0
  50. package/dist/generated/prisma/models/PlanLimit.js +1 -0
  51. package/dist/generated/prisma/models/PodcastSegment.d.ts +1564 -0
  52. package/dist/generated/prisma/models/PodcastSegment.js +1 -0
  53. package/dist/generated/prisma/models/ResourcePrice.d.ts +1008 -0
  54. package/dist/generated/prisma/models/ResourcePrice.js +1 -0
  55. package/dist/generated/prisma/models/Role.d.ts +1065 -0
  56. package/dist/generated/prisma/models/Role.js +1 -0
  57. package/dist/generated/prisma/models/Session.d.ts +1105 -0
  58. package/dist/generated/prisma/models/Session.js +1 -0
  59. package/dist/generated/prisma/models/StripeEvent.d.ts +1081 -0
  60. package/dist/generated/prisma/models/StripeEvent.js +1 -0
  61. package/dist/generated/prisma/models/StudyGuideComment.d.ts +1321 -0
  62. package/dist/generated/prisma/models/StudyGuideComment.js +1 -0
  63. package/dist/generated/prisma/models/StudyGuideHighlight.d.ts +1629 -0
  64. package/dist/generated/prisma/models/StudyGuideHighlight.js +1 -0
  65. package/dist/generated/prisma/models/Subscription.d.ts +1677 -0
  66. package/dist/generated/prisma/models/Subscription.js +1 -0
  67. package/dist/generated/prisma/models/User.d.ts +7559 -0
  68. package/dist/generated/prisma/models/User.js +1 -0
  69. package/dist/generated/prisma/models/UserCredit.d.ts +1249 -0
  70. package/dist/generated/prisma/models/UserCredit.js +1 -0
  71. package/dist/generated/prisma/models/VerificationToken.d.ts +946 -0
  72. package/dist/generated/prisma/models/VerificationToken.js +1 -0
  73. package/dist/generated/prisma/models/WorksheetPreset.d.ts +1433 -0
  74. package/dist/generated/prisma/models/WorksheetPreset.js +1 -0
  75. package/dist/generated/prisma/models/WorksheetQuestion.d.ts +1491 -0
  76. package/dist/generated/prisma/models/WorksheetQuestion.js +1 -0
  77. package/dist/generated/prisma/models/WorksheetQuestionProgress.d.ts +1620 -0
  78. package/dist/generated/prisma/models/WorksheetQuestionProgress.js +1 -0
  79. package/dist/generated/prisma/models/Workspace.d.ts +3620 -0
  80. package/dist/generated/prisma/models/Workspace.js +1 -0
  81. package/dist/generated/prisma/models/WorkspaceInvitation.d.ts +1490 -0
  82. package/dist/generated/prisma/models/WorkspaceInvitation.js +1 -0
  83. package/dist/generated/prisma/models/WorkspaceKnowledgeBase.d.ts +1410 -0
  84. package/dist/generated/prisma/models/WorkspaceKnowledgeBase.js +1 -0
  85. package/dist/generated/prisma/models/WorkspaceMember.d.ts +1326 -0
  86. package/dist/generated/prisma/models/WorkspaceMember.js +1 -0
  87. package/dist/generated/prisma/models.d.ts +39 -0
  88. package/dist/generated/prisma/models.js +1 -0
  89. package/dist/src/context.d.ts +27 -0
  90. package/dist/src/context.js +33 -0
  91. package/dist/src/index.d.ts +3 -0
  92. package/dist/src/index.js +1 -0
  93. package/dist/src/lib/ai/config.d.ts +20 -0
  94. package/dist/src/lib/ai/config.js +31 -0
  95. package/dist/src/lib/ai/embedding-client.d.ts +8 -0
  96. package/dist/src/lib/ai/embedding-client.js +30 -0
  97. package/dist/src/lib/ai/index.d.ts +48 -0
  98. package/dist/src/lib/ai/index.js +29 -0
  99. package/dist/src/lib/ai/inference-backend/client.d.ts +28 -0
  100. package/dist/src/lib/ai/inference-backend/client.js +301 -0
  101. package/dist/src/lib/ai/inference-backend/mocks.d.ts +12 -0
  102. package/dist/src/lib/ai/inference-backend/mocks.js +133 -0
  103. package/dist/src/lib/ai/inference-backend/types.d.ts +44 -0
  104. package/dist/src/lib/ai/inference-backend/types.js +1 -0
  105. package/dist/src/lib/ai/json-parse.d.ts +2 -0
  106. package/dist/src/lib/ai/json-parse.js +34 -0
  107. package/dist/src/lib/ai/llm-client.d.ts +7 -0
  108. package/dist/src/lib/ai/llm-client.js +36 -0
  109. package/dist/src/lib/ai/mock.d.ts +2 -0
  110. package/dist/src/lib/ai/mock.js +10 -0
  111. package/dist/src/lib/ai/types.d.ts +9 -0
  112. package/dist/src/lib/ai/types.js +1 -0
  113. package/dist/src/lib/auth.d.ts +36 -0
  114. package/dist/src/lib/auth.js +117 -0
  115. package/dist/src/lib/chunking.d.ts +19 -0
  116. package/dist/src/lib/chunking.js +47 -0
  117. package/dist/src/lib/constants.d.ts +13 -0
  118. package/dist/src/lib/constants.js +12 -0
  119. package/dist/src/lib/curated-kb-seed.d.ts +12 -0
  120. package/dist/src/lib/curated-kb-seed.js +155 -0
  121. package/dist/src/lib/email.d.ts +11 -0
  122. package/dist/src/lib/email.js +152 -0
  123. package/dist/src/lib/embeddings.d.ts +2 -0
  124. package/dist/src/lib/embeddings.js +1 -0
  125. package/dist/src/lib/ensure-curated-kb-catalog.d.ts +6 -0
  126. package/dist/src/lib/ensure-curated-kb-catalog.js +53 -0
  127. package/dist/src/lib/env.d.ts +41 -0
  128. package/dist/src/lib/env.js +57 -0
  129. package/dist/src/lib/errors.d.ts +33 -0
  130. package/dist/src/lib/errors.js +78 -0
  131. package/dist/src/lib/file.d.ts +0 -0
  132. package/dist/src/lib/file.js +1 -0
  133. package/dist/src/lib/inference.d.ts +1 -0
  134. package/dist/src/lib/inference.js +1 -0
  135. package/dist/src/lib/kb-meta.d.ts +8 -0
  136. package/dist/src/lib/kb-meta.js +77 -0
  137. package/dist/src/lib/logger.d.ts +62 -0
  138. package/dist/src/lib/logger.js +364 -0
  139. package/dist/src/lib/pdf.d.ts +11 -0
  140. package/dist/src/lib/pdf.js +11 -0
  141. package/dist/src/lib/prisma.d.ts +3 -0
  142. package/dist/src/lib/prisma.js +15 -0
  143. package/dist/src/lib/pusher.d.ts +38 -0
  144. package/dist/src/lib/pusher.js +170 -0
  145. package/dist/src/lib/retry.d.ts +15 -0
  146. package/dist/src/lib/retry.js +37 -0
  147. package/dist/src/lib/storage.d.ts +11 -0
  148. package/dist/src/lib/storage.js +71 -0
  149. package/dist/src/lib/stripe.d.ts +10 -0
  150. package/dist/src/lib/stripe.js +36 -0
  151. package/dist/src/lib/validation.d.ts +51 -0
  152. package/dist/src/lib/validation.js +64 -0
  153. package/dist/src/lib/workspace-kb.d.ts +5 -0
  154. package/dist/src/lib/workspace-kb.js +7 -0
  155. package/dist/src/repositories/artifact.repository.d.ts +64 -0
  156. package/dist/src/repositories/artifact.repository.js +40 -0
  157. package/dist/src/repositories/base.repository.d.ts +14 -0
  158. package/dist/src/repositories/base.repository.js +14 -0
  159. package/dist/src/repositories/invitation.repository.d.ts +104 -0
  160. package/dist/src/repositories/invitation.repository.js +44 -0
  161. package/dist/src/repositories/notification.repository.d.ts +76 -0
  162. package/dist/src/repositories/notification.repository.js +44 -0
  163. package/dist/src/repositories/user.repository.d.ts +84 -0
  164. package/dist/src/repositories/user.repository.js +37 -0
  165. package/dist/src/repositories/workspace-member.repository.d.ts +35 -0
  166. package/dist/src/repositories/workspace-member.repository.js +31 -0
  167. package/dist/src/repositories/workspace.repository.d.ts +101 -0
  168. package/dist/src/repositories/workspace.repository.js +79 -0
  169. package/dist/src/routers/_app.d.ts +3464 -0
  170. package/dist/src/routers/_app.js +36 -0
  171. package/dist/src/routers/admin.d.ts +358 -0
  172. package/dist/src/routers/admin.js +105 -0
  173. package/dist/src/routers/annotations.d.ts +219 -0
  174. package/dist/src/routers/annotations.js +29 -0
  175. package/dist/src/routers/artifactVersions.d.ts +65 -0
  176. package/dist/src/routers/artifactVersions.js +14 -0
  177. package/dist/src/routers/auth.d.ts +161 -0
  178. package/dist/src/routers/auth.js +97 -0
  179. package/dist/src/routers/chat.d.ts +170 -0
  180. package/dist/src/routers/chat.js +32 -0
  181. package/dist/src/routers/copilot.d.ts +200 -0
  182. package/dist/src/routers/copilot.js +52 -0
  183. package/dist/src/routers/flashcards.d.ts +336 -0
  184. package/dist/src/routers/flashcards.js +93 -0
  185. package/dist/src/routers/knowledgeBase.d.ts +421 -0
  186. package/dist/src/routers/knowledgeBase.js +118 -0
  187. package/dist/src/routers/members.d.ts +169 -0
  188. package/dist/src/routers/members.js +47 -0
  189. package/dist/src/routers/notifications.d.ts +99 -0
  190. package/dist/src/routers/notifications.js +25 -0
  191. package/dist/src/routers/payment.d.ts +80 -0
  192. package/dist/src/routers/payment.js +21 -0
  193. package/dist/src/routers/podcast.d.ts +287 -0
  194. package/dist/src/routers/podcast.js +34 -0
  195. package/dist/src/routers/studyguide.d.ts +36 -0
  196. package/dist/src/routers/studyguide.js +8 -0
  197. package/dist/src/routers/worksheets.d.ts +429 -0
  198. package/dist/src/routers/worksheets.js +139 -0
  199. package/dist/src/routers/workspace.d.ts +563 -0
  200. package/dist/src/routers/workspace.js +104 -0
  201. package/dist/src/scripts/purge-deleted-users.d.ts +1 -0
  202. package/dist/src/scripts/purge-deleted-users.js +148 -0
  203. package/dist/src/server.d.ts +1 -0
  204. package/dist/src/server.js +190 -0
  205. package/dist/src/services/activity/activity-human-description.service.d.ts +13 -0
  206. package/dist/src/services/activity/activity-human-description.service.js +221 -0
  207. package/dist/src/services/activity/activity-human-description.service.test.d.ts +1 -0
  208. package/dist/src/services/activity/activity-human-description.service.test.js +16 -0
  209. package/dist/src/services/activity/activity-log.service.d.ts +87 -0
  210. package/dist/src/services/activity/activity-log.service.js +276 -0
  211. package/dist/src/services/activity/activity-log.service.test.d.ts +1 -0
  212. package/dist/src/services/activity/activity-log.service.test.js +27 -0
  213. package/dist/src/services/admin/admin.service.d.ts +270 -0
  214. package/dist/src/services/admin/admin.service.js +476 -0
  215. package/dist/src/services/ai/ai-session.service.d.ts +5 -0
  216. package/dist/src/services/ai/ai-session.service.js +4 -0
  217. package/dist/src/services/artifacts/annotation.service.d.ts +177 -0
  218. package/dist/src/services/artifacts/annotation.service.js +154 -0
  219. package/dist/src/services/artifacts/artifact-version.service.d.ts +38 -0
  220. package/dist/src/services/artifacts/artifact-version.service.js +129 -0
  221. package/dist/src/services/artifacts/chat.service.d.ts +127 -0
  222. package/dist/src/services/artifacts/chat.service.js +182 -0
  223. package/dist/src/services/artifacts/study-guide.service.d.ts +18 -0
  224. package/dist/src/services/artifacts/study-guide.service.js +65 -0
  225. package/dist/src/services/auth/auth.service.d.ts +94 -0
  226. package/dist/src/services/auth/auth.service.js +368 -0
  227. package/dist/src/services/base.service.d.ts +14 -0
  228. package/dist/src/services/base.service.js +14 -0
  229. package/dist/src/services/billing/payment.service.d.ts +44 -0
  230. package/dist/src/services/billing/payment.service.js +365 -0
  231. package/dist/src/services/billing/subscription.service.d.ts +37 -0
  232. package/dist/src/services/billing/subscription.service.js +654 -0
  233. package/dist/src/services/billing/usage.service.d.ts +47 -0
  234. package/dist/src/services/billing/usage.service.js +149 -0
  235. package/dist/src/services/content/copilot.service.d.ts +113 -0
  236. package/dist/src/services/content/copilot.service.js +439 -0
  237. package/dist/src/services/content/flashcard-progress.service.d.ts +159 -0
  238. package/dist/src/services/content/flashcard-progress.service.js +432 -0
  239. package/dist/src/services/content/flashcard.service.d.ts +184 -0
  240. package/dist/src/services/content/flashcard.service.js +339 -0
  241. package/dist/src/services/content/media-analysis.service.d.ts +23 -0
  242. package/dist/src/services/content/media-analysis.service.js +404 -0
  243. package/dist/src/services/content/podcast.service.d.ts +267 -0
  244. package/dist/src/services/content/podcast.service.js +653 -0
  245. package/dist/src/services/content/worksheet-content.service.d.ts +37 -0
  246. package/dist/src/services/content/worksheet-content.service.js +84 -0
  247. package/dist/src/services/content/worksheet-content.service.test.d.ts +1 -0
  248. package/dist/src/services/content/worksheet-content.service.test.js +69 -0
  249. package/dist/src/services/content/worksheet-generation.service.d.ts +91 -0
  250. package/dist/src/services/content/worksheet-generation.service.js +95 -0
  251. package/dist/src/services/content/worksheet-generation.service.test.d.ts +1 -0
  252. package/dist/src/services/content/worksheet-generation.service.test.js +20 -0
  253. package/dist/src/services/content/worksheet.service.d.ts +347 -0
  254. package/dist/src/services/content/worksheet.service.js +599 -0
  255. package/dist/src/services/knowledge/knowledge-base.service.d.ts +316 -0
  256. package/dist/src/services/knowledge/knowledge-base.service.js +544 -0
  257. package/dist/src/services/members/invitation.service.d.ts +66 -0
  258. package/dist/src/services/members/invitation.service.js +348 -0
  259. package/dist/src/services/members/member.service.d.ts +36 -0
  260. package/dist/src/services/members/member.service.js +193 -0
  261. package/dist/src/services/notifications/notification.service.d.ts +214 -0
  262. package/dist/src/services/notifications/notification.service.js +550 -0
  263. package/dist/src/services/notifications/notification.service.test.d.ts +1 -0
  264. package/dist/src/services/notifications/notification.service.test.js +87 -0
  265. package/dist/src/services/workspace/workspace-analytics.service.d.ts +24 -0
  266. package/dist/src/services/workspace/workspace-analytics.service.js +95 -0
  267. package/dist/src/services/workspace/workspace-kb.service.d.ts +40 -0
  268. package/dist/src/services/workspace/workspace-kb.service.js +184 -0
  269. package/dist/src/services/workspace/workspace.service.d.ts +263 -0
  270. package/dist/src/services/workspace/workspace.service.js +401 -0
  271. package/dist/src/trpc.d.ts +60 -0
  272. package/dist/src/trpc.js +217 -0
  273. package/dist/src/types/index.d.ts +126 -0
  274. package/dist/src/types/index.js +1 -0
  275. package/package.json +8 -9
  276. package/prisma/schema.prisma +3 -4
  277. package/prisma/seed.mjs +5 -2
  278. package/prisma.config.ts +16 -0
  279. package/src/lib/prisma.ts +18 -9
  280. package/src/scripts/purge-deleted-users.ts +1 -3
  281. package/tsconfig.json +3 -0
@@ -0,0 +1,404 @@
1
+ import { TRPCError } from '@trpc/server';
2
+ import { BaseService } from '../base.service.js';
3
+ import { ArtifactType } from '../../lib/constants.js';
4
+ import { supabaseClient } from '../../lib/storage.js';
5
+ import PusherService from '../../lib/pusher.js';
6
+ import { ai } from '../../lib/ai/index.js';
7
+ import { workspaceKbService } from '../workspace/workspace-kb.service.js';
8
+ import { getUserUsage, getUserPlanLimits } from '../billing/usage.service.js';
9
+ import { notifyArtifactFailed, notifyArtifactReady } from '../notifications/notification.service.js';
10
+ import { FlashcardService } from './flashcard.service.js';
11
+ const PIPELINE_STEPS = ['fileUpload', 'fileAnalysis', 'studyGuide', 'flashcards'];
12
+ const PROGRESS_FILENAME_MAX_CHARS = 20;
13
+ function truncateProgressFilename(value, maxChars = PROGRESS_FILENAME_MAX_CHARS) {
14
+ if (value.length <= maxChars)
15
+ return value;
16
+ return `${value.slice(0, Math.max(0, maxChars - 3))}...`;
17
+ }
18
+ function buildCombinedFilenameLabel(fileNames) {
19
+ return truncateProgressFilename(fileNames.join(', '));
20
+ }
21
+ function buildProgressSteps(currentStep, currentStatus, config, overrides) {
22
+ const stepIndex = PIPELINE_STEPS.indexOf(currentStep);
23
+ const steps = {};
24
+ for (let i = 0; i < PIPELINE_STEPS.length; i++) {
25
+ const step = PIPELINE_STEPS[i];
26
+ let status;
27
+ if (overrides?.[step]) {
28
+ status = overrides[step];
29
+ }
30
+ else if (i < stepIndex) {
31
+ status = 'completed';
32
+ }
33
+ else if (i === stepIndex) {
34
+ status = currentStatus;
35
+ }
36
+ else {
37
+ if (step === 'studyGuide' && !config.generateStudyGuide) {
38
+ status = 'skipped';
39
+ }
40
+ else if (step === 'flashcards' && !config.generateFlashcards) {
41
+ status = 'skipped';
42
+ }
43
+ else {
44
+ status = 'pending';
45
+ }
46
+ }
47
+ steps[step] = { order: i + 1, status };
48
+ }
49
+ return steps;
50
+ }
51
+ function buildProgress(status, filename, fileType, currentStep, currentStepStatus, config, extra) {
52
+ return {
53
+ status,
54
+ filename,
55
+ fileType,
56
+ startedAt: new Date().toISOString(),
57
+ steps: buildProgressSteps(currentStep, currentStepStatus, config, extra),
58
+ ...extra,
59
+ };
60
+ }
61
+ export class MediaAnalysisService extends BaseService {
62
+ constructor(db) {
63
+ super(db);
64
+ }
65
+ async updateAnalysisProgress(workspaceId, progress) {
66
+ await this.db.workspace.update({
67
+ where: { id: workspaceId },
68
+ data: { analysisProgress: progress },
69
+ });
70
+ await PusherService.emitAnalysisProgress(workspaceId, progress);
71
+ }
72
+ async uploadAndAnalyzeMedia(input, userId) {
73
+ const workspace = await this.db.workspace.findFirst({
74
+ where: { id: input.workspaceId, ownerId: userId },
75
+ });
76
+ if (!workspace) {
77
+ this.logger.error('Workspace not found', { workspaceId: input.workspaceId, userId });
78
+ throw new TRPCError({ code: 'NOT_FOUND' });
79
+ }
80
+ if (workspace.fileBeingAnalyzed) {
81
+ throw new TRPCError({
82
+ code: 'CONFLICT',
83
+ message: 'File analysis is already in progress for this workspace. Please wait for it to complete.',
84
+ });
85
+ }
86
+ const files = await this.db.fileAsset.findMany({
87
+ where: {
88
+ id: { in: input.files.map((file) => file.id) },
89
+ workspaceId: input.workspaceId,
90
+ userId,
91
+ },
92
+ });
93
+ if (files.length === 0) {
94
+ throw new TRPCError({
95
+ code: 'NOT_FOUND',
96
+ message: 'No files found with the provided IDs',
97
+ });
98
+ }
99
+ for (const file of files) {
100
+ if (!file.bucket || !file.objectKey) {
101
+ throw new TRPCError({
102
+ code: 'BAD_REQUEST',
103
+ message: `File ${file.id} does not have bucket or objectKey set`,
104
+ });
105
+ }
106
+ }
107
+ const primaryFile = files[0];
108
+ const fileType = primaryFile.mimeType.startsWith('image/') ? 'image' : 'pdf';
109
+ const combinedFilenameLabel = buildCombinedFilenameLabel(files.map((file) => file.name));
110
+ try {
111
+ await this.db.workspace.update({
112
+ where: { id: input.workspaceId },
113
+ data: { fileBeingAnalyzed: true },
114
+ });
115
+ const genConfig = {
116
+ generateStudyGuide: input.generateStudyGuide,
117
+ generateFlashcards: input.generateFlashcards,
118
+ };
119
+ PusherService.emitAnalysisProgress(input.workspaceId, buildProgress('starting', primaryFile.name, fileType, 'fileUpload', 'pending', genConfig));
120
+ try {
121
+ await this.updateAnalysisProgress(input.workspaceId, buildProgress('starting', primaryFile.name, fileType, 'fileUpload', 'pending', genConfig));
122
+ }
123
+ catch (error) {
124
+ this.logger.error('Failed to update analysis progress:', error);
125
+ await this.db.workspace.update({
126
+ where: { id: input.workspaceId },
127
+ data: { fileBeingAnalyzed: false },
128
+ });
129
+ await PusherService.emitError(input.workspaceId, `Failed to update analysis progress: ${error}`, 'file_analysis');
130
+ throw error;
131
+ }
132
+ await this.updateAnalysisProgress(input.workspaceId, buildProgress('uploading', primaryFile.name, fileType, 'fileUpload', 'in_progress', genConfig));
133
+ for (const [fileIndex, file] of files.entries()) {
134
+ if (!file.bucket || !file.objectKey) {
135
+ continue;
136
+ }
137
+ const currentFileType = file.mimeType.startsWith('image/') ? 'image' : 'pdf';
138
+ const currentFileLabel = truncateProgressFilename(file.name);
139
+ await this.updateAnalysisProgress(input.workspaceId, buildProgress('uploading', currentFileLabel, currentFileType, 'fileUpload', 'in_progress', genConfig, {
140
+ currentFileIndex: fileIndex + 1,
141
+ totalFiles: files.length,
142
+ }));
143
+ const { data: signedUrlData, error: signedUrlError } = await supabaseClient.storage
144
+ .from(file.bucket)
145
+ .createSignedUrl(file.objectKey, 24 * 60 * 60);
146
+ if (signedUrlError) {
147
+ await this.db.workspace.update({
148
+ where: { id: input.workspaceId },
149
+ data: { fileBeingAnalyzed: false },
150
+ });
151
+ throw new TRPCError({
152
+ code: 'INTERNAL_SERVER_ERROR',
153
+ message: `Failed to upload file`,
154
+ });
155
+ }
156
+ const fileUrl = signedUrlData.signedUrl;
157
+ const maxPages = currentFileType === 'pdf' && file.size && file.size > 50 ? 50 : undefined;
158
+ const processResult = await ai.backend.processFile(input.workspaceId, userId, fileUrl, currentFileType, maxPages);
159
+ if (processResult.status === 'error') {
160
+ this.logger.error(`Failed to process file ${file.name}:`, processResult.error);
161
+ }
162
+ else {
163
+ this.logger.info(`Successfully processed file ${file.name}: ${processResult.pageCount} pages`);
164
+ await this.db.fileAsset.update({
165
+ where: { id: file.id },
166
+ data: {
167
+ aiTranscription: {
168
+ comprehensiveDescription: processResult.comprehensiveDescription,
169
+ textContent: processResult.textContent,
170
+ imageDescriptions: processResult.imageDescriptions,
171
+ },
172
+ },
173
+ });
174
+ const indexText = processResult.comprehensiveDescription ??
175
+ processResult.textContent ??
176
+ '';
177
+ if (indexText.trim()) {
178
+ workspaceKbService.indexTextAsync(input.workspaceId, userId, {
179
+ sourceType: 'file',
180
+ sourceId: file.id,
181
+ name: file.name,
182
+ text: indexText,
183
+ });
184
+ }
185
+ }
186
+ }
187
+ await this.updateAnalysisProgress(input.workspaceId, buildProgress('analyzing', combinedFilenameLabel, fileType, 'fileAnalysis', 'in_progress', genConfig));
188
+ try {
189
+ await this.updateAnalysisProgress(input.workspaceId, buildProgress('generating_artifacts', combinedFilenameLabel, fileType, 'studyGuide', 'pending', genConfig));
190
+ }
191
+ catch (error) {
192
+ this.logger.error('Failed to analyze files:', error);
193
+ await this.updateAnalysisProgress(input.workspaceId, buildProgress('error', combinedFilenameLabel, fileType, 'fileAnalysis', 'error', genConfig, {
194
+ error: `Failed to analyze ${fileType}: ${error}`,
195
+ studyGuide: 'skipped',
196
+ flashcards: 'skipped',
197
+ }));
198
+ await this.db.workspace.update({
199
+ where: { id: input.workspaceId },
200
+ data: { fileBeingAnalyzed: false },
201
+ });
202
+ throw error;
203
+ }
204
+ const results = {
205
+ filename: primaryFile.name,
206
+ artifacts: {
207
+ studyGuide: null,
208
+ flashcards: null,
209
+ worksheet: null,
210
+ },
211
+ };
212
+ try {
213
+ await ai.backend.initSession(input.workspaceId, userId);
214
+ }
215
+ catch (initError) {
216
+ this.logger.error('Failed to init AI session (continuing with workspace context):', initError);
217
+ }
218
+ const [usage, limits] = await Promise.all([
219
+ getUserUsage(userId),
220
+ getUserPlanLimits(userId),
221
+ ]);
222
+ if (input.generateStudyGuide) {
223
+ if (limits && usage.studyGuides >= limits.maxStudyGuides) {
224
+ await this.updateAnalysisProgress(input.workspaceId, buildProgress('skipped', combinedFilenameLabel, fileType, 'studyGuide', 'skipped', genConfig));
225
+ await PusherService.emitError(input.workspaceId, 'Study guide skipped: Limit reached.', 'study_guide');
226
+ await notifyArtifactFailed(this.db, {
227
+ userId,
228
+ workspaceId: input.workspaceId,
229
+ artifactType: ArtifactType.STUDY_GUIDE,
230
+ message: 'Study guide was skipped because your plan limit was reached.',
231
+ }).catch(() => { });
232
+ }
233
+ else {
234
+ try {
235
+ await this.updateAnalysisProgress(input.workspaceId, buildProgress('generating_study_guide', combinedFilenameLabel, fileType, 'studyGuide', 'in_progress', genConfig));
236
+ const ragContext = await workspaceKbService.retrieveContext(input.workspaceId, 'study guide key concepts definitions explanations examples', 8);
237
+ const content = await ai.backend.generateStudyGuide(input.workspaceId, userId, { ragContext });
238
+ let artifact = await this.db.artifact.findFirst({
239
+ where: { workspaceId: input.workspaceId, type: ArtifactType.STUDY_GUIDE },
240
+ });
241
+ if (!artifact) {
242
+ artifact = await this.db.artifact.create({
243
+ data: {
244
+ workspaceId: input.workspaceId,
245
+ type: ArtifactType.STUDY_GUIDE,
246
+ title: files.length === 1
247
+ ? `Study Guide - ${primaryFile.name}`
248
+ : `Study Guide - ${files.length} files`,
249
+ createdById: userId,
250
+ },
251
+ });
252
+ }
253
+ const lastVersion = await this.db.artifactVersion.findFirst({
254
+ where: { artifact: { workspaceId: input.workspaceId, type: ArtifactType.STUDY_GUIDE } },
255
+ orderBy: { version: 'desc' },
256
+ });
257
+ await this.db.artifactVersion.create({
258
+ data: {
259
+ artifactId: artifact.id,
260
+ version: lastVersion ? lastVersion.version + 1 : 1,
261
+ content: content,
262
+ createdById: userId,
263
+ },
264
+ });
265
+ results.artifacts.studyGuide = artifact;
266
+ await PusherService.emitStudyGuideComplete(input.workspaceId, artifact);
267
+ await notifyArtifactReady(this.db, {
268
+ userId,
269
+ workspaceId: input.workspaceId,
270
+ artifactId: artifact.id,
271
+ artifactType: ArtifactType.STUDY_GUIDE,
272
+ title: artifact.title,
273
+ }).catch(() => { });
274
+ workspaceKbService.indexTextAsync(input.workspaceId, userId, {
275
+ sourceType: 'study-guide',
276
+ sourceId: artifact.id,
277
+ name: artifact.title,
278
+ text: content,
279
+ });
280
+ }
281
+ catch (sgError) {
282
+ this.logger.error('Study guide generation failed after retries:', sgError);
283
+ await PusherService.emitError(input.workspaceId, 'Study guide generation failed. Please try regenerating later.', 'study_guide');
284
+ await notifyArtifactFailed(this.db, {
285
+ userId,
286
+ workspaceId: input.workspaceId,
287
+ artifactType: ArtifactType.STUDY_GUIDE,
288
+ message: 'Study guide generation failed. Please try regenerating later.',
289
+ }).catch(() => { });
290
+ }
291
+ }
292
+ }
293
+ if (input.generateFlashcards) {
294
+ if (limits && usage.flashcards >= limits.maxFlashcards) {
295
+ await this.updateAnalysisProgress(input.workspaceId, buildProgress('skipped', combinedFilenameLabel, fileType, 'flashcards', 'skipped', genConfig));
296
+ await PusherService.emitError(input.workspaceId, 'Flashcards skipped: Limit reached.', 'flashcards');
297
+ await notifyArtifactFailed(this.db, {
298
+ userId,
299
+ workspaceId: input.workspaceId,
300
+ artifactType: ArtifactType.FLASHCARD_SET,
301
+ message: 'Flashcards were skipped because your plan limit was reached.',
302
+ }).catch(() => { });
303
+ }
304
+ else {
305
+ try {
306
+ const sgStatus = input.generateStudyGuide
307
+ ? results.artifacts.studyGuide
308
+ ? 'completed'
309
+ : 'error'
310
+ : 'skipped';
311
+ await this.updateAnalysisProgress(input.workspaceId, buildProgress('generating_flashcards', combinedFilenameLabel, fileType, 'flashcards', 'in_progress', genConfig, {
312
+ studyGuide: sgStatus,
313
+ }));
314
+ const flashcardRag = await workspaceKbService.retrieveContext(input.workspaceId, 'flashcard key concepts facts definitions questions', 8);
315
+ const content = await ai.backend.generateFlashcardQuestions(input.workspaceId, userId, 10, 'medium', undefined, { ragContext: flashcardRag });
316
+ const flashcardService = new FlashcardService(this.db);
317
+ const primarySet = await flashcardService.ensurePrimarySet(userId, input.workspaceId);
318
+ const orderOffset = primarySet.flashcards.reduce((max, card) => Math.max(max, card.order), -1);
319
+ const flashcardTitle = files.length === 1
320
+ ? `Flashcards - ${primaryFile.name}`
321
+ : `Flashcards - ${files.length} files`;
322
+ await this.db.artifact.update({
323
+ where: { id: primarySet.id },
324
+ data: { title: flashcardTitle },
325
+ });
326
+ const appendCard = async (front, back, index) => {
327
+ await this.db.flashcard.create({
328
+ data: {
329
+ artifactId: primarySet.id,
330
+ front,
331
+ back,
332
+ order: orderOffset + 1 + index,
333
+ tags: ['ai-generated', 'medium'],
334
+ },
335
+ });
336
+ };
337
+ try {
338
+ const parsed = typeof content === 'string' ? JSON.parse(content) : content;
339
+ const flashcardData = Array.isArray(parsed) ? parsed : (parsed.flashcards || []);
340
+ for (let i = 0; i < Math.min(flashcardData.length, 10); i++) {
341
+ const card = flashcardData[i];
342
+ const front = card.term || card.front || card.question || card.prompt || `Question ${i + 1}`;
343
+ const back = card.definition || card.back || card.answer || card.solution || `Answer ${i + 1}`;
344
+ await appendCard(front, back, i);
345
+ }
346
+ }
347
+ catch (parseError) {
348
+ console.error('Failed to parse flashcard JSON or create cards in workspace router:', parseError);
349
+ const lines = content.split('\n').filter((line) => line.trim());
350
+ for (let i = 0; i < Math.min(lines.length, 10); i++) {
351
+ const line = lines[i];
352
+ if (line.includes(' - ')) {
353
+ const [front, back] = line.split(' - ');
354
+ await appendCard(front.trim(), back.trim(), i);
355
+ }
356
+ }
357
+ }
358
+ const artifact = await this.db.artifact.findUniqueOrThrow({
359
+ where: { id: primarySet.id },
360
+ include: { flashcards: true },
361
+ });
362
+ results.artifacts.flashcards = artifact;
363
+ await PusherService.emitFlashcardComplete(input.workspaceId, artifact);
364
+ await notifyArtifactReady(this.db, {
365
+ userId,
366
+ workspaceId: input.workspaceId,
367
+ artifactId: artifact.id,
368
+ artifactType: ArtifactType.FLASHCARD_SET,
369
+ title: artifact.title,
370
+ }).catch(() => { });
371
+ }
372
+ catch (fcError) {
373
+ this.logger.error('Flashcard generation failed after retries:', fcError);
374
+ await PusherService.emitError(input.workspaceId, 'Flashcard generation failed. Please try regenerating later.', 'flashcards');
375
+ await notifyArtifactFailed(this.db, {
376
+ userId,
377
+ workspaceId: input.workspaceId,
378
+ artifactType: ArtifactType.FLASHCARD_SET,
379
+ message: 'Flashcard generation failed. Please try regenerating later.',
380
+ }).catch(() => { });
381
+ }
382
+ }
383
+ }
384
+ await this.db.workspace.update({
385
+ where: { id: input.workspaceId },
386
+ data: { fileBeingAnalyzed: false },
387
+ });
388
+ await this.updateAnalysisProgress(input.workspaceId, {
389
+ ...buildProgress('completed', combinedFilenameLabel, fileType, 'flashcards', 'completed', genConfig),
390
+ completedAt: new Date().toISOString(),
391
+ });
392
+ return results;
393
+ }
394
+ catch (error) {
395
+ this.logger.error('Failed to update analysis progress:', error);
396
+ await this.db.workspace.update({
397
+ where: { id: input.workspaceId },
398
+ data: { fileBeingAnalyzed: false },
399
+ });
400
+ await PusherService.emitError(input.workspaceId, `Failed to update analysis progress: ${error}`, 'file_analysis');
401
+ throw error;
402
+ }
403
+ }
404
+ }
@@ -0,0 +1,267 @@
1
+ import type { PrismaClient } from '@prisma/client';
2
+ import { z } from 'zod';
3
+ import { BaseService } from '../base.service.js';
4
+ export declare const speakerSchema: z.ZodObject<{
5
+ id: z.ZodString;
6
+ role: z.ZodEnum<{
7
+ host: "host";
8
+ guest: "guest";
9
+ expert: "expert";
10
+ }>;
11
+ name: z.ZodOptional<z.ZodString>;
12
+ }, z.core.$strip>;
13
+ export declare const podcastInputSchema: z.ZodObject<{
14
+ title: z.ZodString;
15
+ description: z.ZodOptional<z.ZodString>;
16
+ userPrompt: z.ZodString;
17
+ speakers: z.ZodDefault<z.ZodArray<z.ZodObject<{
18
+ id: z.ZodString;
19
+ role: z.ZodEnum<{
20
+ host: "host";
21
+ guest: "guest";
22
+ expert: "expert";
23
+ }>;
24
+ name: z.ZodOptional<z.ZodString>;
25
+ }, z.core.$strip>>>;
26
+ speed: z.ZodDefault<z.ZodNumber>;
27
+ generateIntro: z.ZodDefault<z.ZodBoolean>;
28
+ generateOutro: z.ZodDefault<z.ZodBoolean>;
29
+ segmentByTopics: z.ZodDefault<z.ZodBoolean>;
30
+ }, z.core.$strip>;
31
+ export type PodcastInput = z.infer<typeof podcastInputSchema>;
32
+ export declare const podcastMetadataSchema: z.ZodObject<{
33
+ title: z.ZodString;
34
+ description: z.ZodOptional<z.ZodString>;
35
+ totalDuration: z.ZodNumber;
36
+ speakers: z.ZodArray<z.ZodObject<{
37
+ id: z.ZodString;
38
+ role: z.ZodEnum<{
39
+ host: "host";
40
+ guest: "guest";
41
+ expert: "expert";
42
+ }>;
43
+ name: z.ZodOptional<z.ZodString>;
44
+ }, z.core.$strip>>;
45
+ summary: z.ZodObject<{
46
+ executiveSummary: z.ZodString;
47
+ learningObjectives: z.ZodArray<z.ZodString>;
48
+ keyConcepts: z.ZodArray<z.ZodString>;
49
+ followUpActions: z.ZodArray<z.ZodString>;
50
+ targetAudience: z.ZodString;
51
+ prerequisites: z.ZodArray<z.ZodString>;
52
+ tags: z.ZodArray<z.ZodString>;
53
+ }, z.core.$strip>;
54
+ generatedAt: z.ZodString;
55
+ }, z.core.$strip>;
56
+ export declare class PodcastService extends BaseService {
57
+ constructor(db: PrismaClient);
58
+ listEpisodes(userId: string, workspaceId: string): Promise<{
59
+ id: string;
60
+ title: string;
61
+ description: string | null;
62
+ metadata: {
63
+ title: string;
64
+ totalDuration: number;
65
+ speakers: {
66
+ id: string;
67
+ role: "host" | "guest" | "expert";
68
+ name?: string | undefined;
69
+ }[];
70
+ summary: {
71
+ executiveSummary: string;
72
+ learningObjectives: string[];
73
+ keyConcepts: string[];
74
+ followUpActions: string[];
75
+ targetAudience: string;
76
+ prerequisites: string[];
77
+ tags: string[];
78
+ };
79
+ generatedAt: string;
80
+ description?: string | undefined;
81
+ } | null;
82
+ imageUrl: string | null;
83
+ segments: ({
84
+ id: string;
85
+ title: string;
86
+ audioUrl: string;
87
+ objectKey: string;
88
+ startTime: number;
89
+ duration: number;
90
+ order: number;
91
+ } | {
92
+ id: string;
93
+ title: string;
94
+ audioUrl: null;
95
+ objectKey: string | null;
96
+ startTime: number;
97
+ duration: number;
98
+ order: number;
99
+ })[];
100
+ createdAt: Date;
101
+ updatedAt: Date;
102
+ workspaceId: string;
103
+ generating: boolean;
104
+ generatingMetadata: import("@prisma/client/runtime/client").JsonValue;
105
+ type: import("@prisma/client").ArtifactType;
106
+ createdById: string | null;
107
+ isArchived: boolean;
108
+ }[]>;
109
+ getEpisode(userId: string, episodeId: string): Promise<{
110
+ id: string;
111
+ title: string;
112
+ description: string | undefined;
113
+ metadata: {
114
+ title: string;
115
+ totalDuration: number;
116
+ speakers: {
117
+ id: string;
118
+ role: "host" | "guest" | "expert";
119
+ name?: string | undefined;
120
+ }[];
121
+ summary: {
122
+ executiveSummary: string;
123
+ learningObjectives: string[];
124
+ keyConcepts: string[];
125
+ followUpActions: string[];
126
+ targetAudience: string;
127
+ prerequisites: string[];
128
+ tags: string[];
129
+ };
130
+ generatedAt: string;
131
+ description?: string | undefined;
132
+ };
133
+ imageUrl: string | null;
134
+ segments: ({
135
+ id: string;
136
+ title: string;
137
+ content: string;
138
+ audioUrl: string;
139
+ objectKey: string;
140
+ startTime: number;
141
+ duration: number;
142
+ keyPoints: string[];
143
+ order: number;
144
+ } | {
145
+ id: string;
146
+ title: string;
147
+ content: string;
148
+ audioUrl: null;
149
+ objectKey: string | null;
150
+ startTime: number;
151
+ duration: number;
152
+ keyPoints: string[];
153
+ order: number;
154
+ })[];
155
+ content: string;
156
+ createdAt: Date;
157
+ updatedAt: Date;
158
+ }>;
159
+ generateEpisode(userId: string, input: {
160
+ workspaceId: string;
161
+ podcastData: PodcastInput;
162
+ }): Promise<{
163
+ id: string;
164
+ title: any;
165
+ description: string | undefined;
166
+ metadata: {
167
+ title: any;
168
+ description: string | undefined;
169
+ totalDuration: number;
170
+ summary: any;
171
+ speakers: {
172
+ id: string;
173
+ role: "host" | "guest" | "expert";
174
+ name?: string | undefined;
175
+ }[];
176
+ generatedAt: string;
177
+ };
178
+ content: string;
179
+ }>;
180
+ deleteSegment(segmentId: string): Promise<{
181
+ id: string;
182
+ createdAt: Date;
183
+ updatedAt: Date;
184
+ title: string;
185
+ duration: number;
186
+ objectKey: string | null;
187
+ artifactId: string;
188
+ content: string;
189
+ meta: import("@prisma/client/runtime/client").JsonValue | null;
190
+ order: number;
191
+ generating: boolean;
192
+ generatingMetadata: import("@prisma/client/runtime/client").JsonValue | null;
193
+ startTime: number;
194
+ audioUrl: string | null;
195
+ keyPoints: string[];
196
+ }>;
197
+ getEpisodeSchema(userId: string, episodeId: string): Promise<{
198
+ segments: {
199
+ id: string;
200
+ title: string;
201
+ startTime: number;
202
+ duration: number;
203
+ keyPoints: string[];
204
+ order: number;
205
+ }[];
206
+ summary: {
207
+ executiveSummary: string;
208
+ learningObjectives: string[];
209
+ keyConcepts: string[];
210
+ followUpActions: string[];
211
+ targetAudience: string;
212
+ prerequisites: string[];
213
+ tags: string[];
214
+ };
215
+ metadata: {
216
+ title: string;
217
+ description: string | undefined;
218
+ totalDuration: number;
219
+ speakers: {
220
+ id: string;
221
+ role: "host" | "guest" | "expert";
222
+ name?: string | undefined;
223
+ }[];
224
+ };
225
+ }>;
226
+ updateEpisode(userId: string, input: {
227
+ episodeId: string;
228
+ title?: string;
229
+ description?: string;
230
+ }): Promise<{
231
+ id: string;
232
+ createdAt: Date;
233
+ updatedAt: Date;
234
+ title: string;
235
+ description: string | null;
236
+ createdById: string | null;
237
+ type: import("@prisma/client").ArtifactType;
238
+ workspaceId: string;
239
+ difficulty: import("@prisma/client").Difficulty | null;
240
+ estimatedTime: string | null;
241
+ isArchived: boolean;
242
+ generating: boolean;
243
+ generatingMetadata: import("@prisma/client/runtime/client").JsonValue | null;
244
+ worksheetConfig: import("@prisma/client/runtime/client").JsonValue | null;
245
+ imageObjectKey: string | null;
246
+ }>;
247
+ deleteEpisode(userId: string, episodeId: string): Promise<boolean>;
248
+ getSegment(userId: string, segmentId: string): Promise<{
249
+ id: string;
250
+ title: string;
251
+ content: string;
252
+ startTime: number;
253
+ duration: number;
254
+ order: number;
255
+ keyPoints: string[];
256
+ audioUrl: string | null;
257
+ objectKey: string | null;
258
+ meta: import("@prisma/client/runtime/client").JsonValue;
259
+ createdAt: Date;
260
+ updatedAt: Date;
261
+ }>;
262
+ getAvailableVoices(): {
263
+ id: string;
264
+ name: string;
265
+ description: string;
266
+ }[];
267
+ }