@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,184 @@
1
+ import type { PrismaClient } from '@prisma/client';
2
+ import { z } from 'zod';
3
+ import { BaseService } from '../base.service.js';
4
+ export declare const typedAnswerGradeSchema: z.ZodObject<{
5
+ isCorrect: z.ZodBoolean;
6
+ confidence: z.ZodNumber;
7
+ reason: z.ZodString;
8
+ matchedAnswer: z.ZodNullable<z.ZodString>;
9
+ }, z.core.$strip>;
10
+ export declare class FlashcardService extends BaseService {
11
+ constructor(db: PrismaClient);
12
+ private findPrimarySet;
13
+ /** Returns the primary set, creating one only when flashcard content is first added. */
14
+ ensurePrimarySet(userId: string, workspaceId: string): Promise<{
15
+ _count: {
16
+ flashcards: number;
17
+ };
18
+ flashcards: {
19
+ id: string;
20
+ createdAt: Date;
21
+ tags: string[];
22
+ artifactId: string;
23
+ order: number;
24
+ front: string;
25
+ back: string;
26
+ acceptedAnswers: string[];
27
+ }[];
28
+ } & {
29
+ id: string;
30
+ createdAt: Date;
31
+ updatedAt: Date;
32
+ title: string;
33
+ description: string | null;
34
+ createdById: string | null;
35
+ type: import("@prisma/client").ArtifactType;
36
+ workspaceId: string;
37
+ difficulty: import("@prisma/client").Difficulty | null;
38
+ estimatedTime: string | null;
39
+ isArchived: boolean;
40
+ generating: boolean;
41
+ generatingMetadata: import("@prisma/client/runtime/client").JsonValue | null;
42
+ worksheetConfig: import("@prisma/client/runtime/client").JsonValue | null;
43
+ imageObjectKey: string | null;
44
+ }>;
45
+ listSets(userId: string, workspaceId: string): Promise<({
46
+ versions: {
47
+ id: string;
48
+ createdAt: Date;
49
+ data: import("@prisma/client/runtime/client").JsonValue | null;
50
+ createdById: string | null;
51
+ artifactId: string;
52
+ content: string;
53
+ version: number;
54
+ }[];
55
+ } & {
56
+ id: string;
57
+ createdAt: Date;
58
+ updatedAt: Date;
59
+ title: string;
60
+ description: string | null;
61
+ createdById: string | null;
62
+ type: import("@prisma/client").ArtifactType;
63
+ workspaceId: string;
64
+ difficulty: import("@prisma/client").Difficulty | null;
65
+ estimatedTime: string | null;
66
+ isArchived: boolean;
67
+ generating: boolean;
68
+ generatingMetadata: import("@prisma/client/runtime/client").JsonValue | null;
69
+ worksheetConfig: import("@prisma/client/runtime/client").JsonValue | null;
70
+ imageObjectKey: string | null;
71
+ })[]>;
72
+ listCards(userId: string, workspaceId: string): Promise<({
73
+ progress: {
74
+ userId: string;
75
+ id: string;
76
+ createdAt: Date;
77
+ updatedAt: Date;
78
+ interval: number;
79
+ flashcardId: string;
80
+ timesStudied: number;
81
+ timesCorrect: number;
82
+ timesIncorrect: number;
83
+ timesIncorrectConsecutive: number;
84
+ easeFactor: number;
85
+ repetitions: number;
86
+ masteryLevel: number;
87
+ lastStudiedAt: Date | null;
88
+ nextReviewAt: Date | null;
89
+ }[];
90
+ } & {
91
+ id: string;
92
+ createdAt: Date;
93
+ tags: string[];
94
+ artifactId: string;
95
+ order: number;
96
+ front: string;
97
+ back: string;
98
+ acceptedAnswers: string[];
99
+ })[]>;
100
+ isGenerating(userId: string, workspaceId: string): Promise<boolean>;
101
+ createCard(userId: string, input: {
102
+ workspaceId: string;
103
+ front: string;
104
+ back: string;
105
+ acceptedAnswers?: string[];
106
+ tags?: string[];
107
+ order?: number;
108
+ }): Promise<{
109
+ id: string;
110
+ createdAt: Date;
111
+ tags: string[];
112
+ artifactId: string;
113
+ order: number;
114
+ front: string;
115
+ back: string;
116
+ acceptedAnswers: string[];
117
+ }>;
118
+ updateCard(userId: string, input: {
119
+ cardId: string;
120
+ front?: string;
121
+ back?: string;
122
+ acceptedAnswers?: string[];
123
+ tags?: string[];
124
+ order?: number;
125
+ }): Promise<{
126
+ id: string;
127
+ createdAt: Date;
128
+ tags: string[];
129
+ artifactId: string;
130
+ order: number;
131
+ front: string;
132
+ back: string;
133
+ acceptedAnswers: string[];
134
+ }>;
135
+ gradeTypedAnswer(userId: string, input: {
136
+ flashcardId: string;
137
+ userAnswer: string;
138
+ }): Promise<{
139
+ isCorrect: boolean;
140
+ confidence: number;
141
+ reason: string;
142
+ matchedAnswer: string | null;
143
+ }>;
144
+ deleteCard(userId: string, cardId: string): Promise<boolean>;
145
+ deleteSet(userId: string, setId: string): Promise<boolean>;
146
+ generateFromPrompt(userId: string, input: {
147
+ workspaceId: string;
148
+ prompt: string;
149
+ numCards: number;
150
+ difficulty: 'easy' | 'medium' | 'hard';
151
+ title?: string;
152
+ tags?: string[];
153
+ }): Promise<{
154
+ artifact: {
155
+ flashcards: {
156
+ id: string;
157
+ createdAt: Date;
158
+ tags: string[];
159
+ artifactId: string;
160
+ order: number;
161
+ front: string;
162
+ back: string;
163
+ acceptedAnswers: string[];
164
+ }[];
165
+ } & {
166
+ id: string;
167
+ createdAt: Date;
168
+ updatedAt: Date;
169
+ title: string;
170
+ description: string | null;
171
+ createdById: string | null;
172
+ type: import("@prisma/client").ArtifactType;
173
+ workspaceId: string;
174
+ difficulty: import("@prisma/client").Difficulty | null;
175
+ estimatedTime: string | null;
176
+ isArchived: boolean;
177
+ generating: boolean;
178
+ generatingMetadata: import("@prisma/client/runtime/client").JsonValue | null;
179
+ worksheetConfig: import("@prisma/client/runtime/client").JsonValue | null;
180
+ imageObjectKey: string | null;
181
+ };
182
+ createdCards: number;
183
+ }>;
184
+ }
@@ -0,0 +1,339 @@
1
+ import { TRPCError } from '@trpc/server';
2
+ import { z } from 'zod';
3
+ import { BaseService } from '../base.service.js';
4
+ import { ArtifactType } from '../../lib/constants.js';
5
+ import { workspaceAccessWhere } from '../../repositories/workspace.repository.js';
6
+ import { ai } from '../../lib/ai/index.js';
7
+ import { workspaceKbService } from '../workspace/workspace-kb.service.js';
8
+ import PusherService from '../../lib/pusher.js';
9
+ import { notifyArtifactFailed, notifyArtifactReady } from '../notifications/notification.service.js';
10
+ import { invalidateUserBillingCache } from '../billing/usage.service.js';
11
+ export const typedAnswerGradeSchema = z.object({
12
+ isCorrect: z.boolean(),
13
+ confidence: z.number().min(0).max(1),
14
+ reason: z.string().min(1),
15
+ matchedAnswer: z.string().nullable(),
16
+ });
17
+ function normalizeAcceptedAnswers(answers) {
18
+ if (!answers || answers.length === 0)
19
+ return [];
20
+ const seen = new Set();
21
+ const normalized = [];
22
+ for (const answer of answers) {
23
+ const trimmed = answer.trim();
24
+ if (!trimmed)
25
+ continue;
26
+ const key = trimmed.toLowerCase();
27
+ if (seen.has(key))
28
+ continue;
29
+ seen.add(key);
30
+ normalized.push(trimmed);
31
+ }
32
+ return normalized;
33
+ }
34
+ function extractFirstJsonObject(text) {
35
+ const start = text.indexOf('{');
36
+ const end = text.lastIndexOf('}');
37
+ if (start === -1 || end === -1 || end <= start)
38
+ return null;
39
+ return text.slice(start, end + 1);
40
+ }
41
+ export class FlashcardService extends BaseService {
42
+ constructor(db) {
43
+ super(db);
44
+ }
45
+ async findPrimarySet(userId, workspaceId) {
46
+ const sets = await this.db.artifact.findMany({
47
+ where: {
48
+ workspaceId,
49
+ type: ArtifactType.FLASHCARD_SET,
50
+ workspace: workspaceAccessWhere(userId),
51
+ },
52
+ include: {
53
+ flashcards: { orderBy: { order: 'asc' } },
54
+ _count: { select: { flashcards: true } },
55
+ },
56
+ orderBy: { createdAt: 'asc' },
57
+ });
58
+ if (sets.length === 0)
59
+ return null;
60
+ const withCards = sets.filter((set) => set._count.flashcards > 0);
61
+ const candidates = withCards.length > 0 ? withCards : sets;
62
+ return candidates.reduce((best, current) => current._count.flashcards > best._count.flashcards ? current : best);
63
+ }
64
+ /** Returns the primary set, creating one only when flashcard content is first added. */
65
+ async ensurePrimarySet(userId, workspaceId) {
66
+ const existing = await this.findPrimarySet(userId, workspaceId);
67
+ if (existing)
68
+ return existing;
69
+ const workspace = await this.db.workspace.findFirst({
70
+ where: { id: workspaceId, ...workspaceAccessWhere(userId) },
71
+ select: { id: true, title: true },
72
+ });
73
+ if (!workspace) {
74
+ throw new TRPCError({ code: 'NOT_FOUND', message: 'Workspace not found' });
75
+ }
76
+ return this.db.artifact.create({
77
+ data: {
78
+ workspaceId,
79
+ type: ArtifactType.FLASHCARD_SET,
80
+ title: 'Flashcards',
81
+ createdById: userId,
82
+ },
83
+ include: {
84
+ flashcards: { orderBy: { order: 'asc' } },
85
+ _count: { select: { flashcards: true } },
86
+ },
87
+ });
88
+ }
89
+ async listSets(userId, workspaceId) {
90
+ const workspace = await this.db.workspace.findFirst({
91
+ where: { id: workspaceId, ownerId: userId },
92
+ });
93
+ if (!workspace)
94
+ throw new TRPCError({ code: 'NOT_FOUND' });
95
+ return this.db.artifact.findMany({
96
+ where: { workspaceId, type: ArtifactType.FLASHCARD_SET },
97
+ include: {
98
+ versions: { orderBy: { version: 'desc' }, take: 1 },
99
+ },
100
+ orderBy: { updatedAt: 'desc' },
101
+ });
102
+ }
103
+ async listCards(userId, workspaceId) {
104
+ const set = await this.findPrimarySet(userId, workspaceId);
105
+ if (!set)
106
+ return [];
107
+ return this.db.flashcard.findMany({
108
+ where: { artifactId: set.id },
109
+ include: {
110
+ progress: { where: { userId } },
111
+ },
112
+ orderBy: { order: 'asc' },
113
+ });
114
+ }
115
+ async isGenerating(userId, workspaceId) {
116
+ const set = await this.findPrimarySet(userId, workspaceId);
117
+ return set?.generating ?? false;
118
+ }
119
+ async createCard(userId, input) {
120
+ const set = await this.ensurePrimarySet(userId, input.workspaceId);
121
+ const nextOrder = input.order ??
122
+ (set.flashcards.reduce((max, card) => Math.max(max, card.order), -1) + 1);
123
+ const card = await this.db.flashcard.create({
124
+ data: {
125
+ artifactId: set.id,
126
+ front: input.front,
127
+ back: input.back,
128
+ acceptedAnswers: normalizeAcceptedAnswers(input.acceptedAnswers),
129
+ tags: input.tags ?? [],
130
+ order: nextOrder,
131
+ },
132
+ });
133
+ invalidateUserBillingCache(userId);
134
+ return card;
135
+ }
136
+ async updateCard(userId, input) {
137
+ const card = await this.db.flashcard.findFirst({
138
+ where: {
139
+ id: input.cardId,
140
+ artifact: {
141
+ type: ArtifactType.FLASHCARD_SET,
142
+ workspace: workspaceAccessWhere(userId),
143
+ },
144
+ },
145
+ });
146
+ if (!card)
147
+ throw new TRPCError({ code: 'NOT_FOUND' });
148
+ return this.db.flashcard.update({
149
+ where: { id: input.cardId },
150
+ data: {
151
+ front: input.front ?? card.front,
152
+ back: input.back ?? card.back,
153
+ acceptedAnswers: input.acceptedAnswers
154
+ ? normalizeAcceptedAnswers(input.acceptedAnswers)
155
+ : card.acceptedAnswers,
156
+ tags: input.tags ?? card.tags,
157
+ order: input.order ?? card.order,
158
+ },
159
+ });
160
+ }
161
+ async gradeTypedAnswer(userId, input) {
162
+ const flashcard = await this.db.flashcard.findFirst({
163
+ where: {
164
+ id: input.flashcardId,
165
+ artifact: {
166
+ type: ArtifactType.FLASHCARD_SET,
167
+ workspace: workspaceAccessWhere(userId),
168
+ },
169
+ },
170
+ select: { id: true, front: true, back: true, acceptedAnswers: true },
171
+ });
172
+ if (!flashcard) {
173
+ throw new TRPCError({ code: 'NOT_FOUND', message: 'Flashcard not found' });
174
+ }
175
+ const acceptedAnswers = [
176
+ flashcard.back,
177
+ ...normalizeAcceptedAnswers(flashcard.acceptedAnswers),
178
+ ];
179
+ const prompt = [
180
+ 'Grade whether the student answer is semantically correct for the flashcard.',
181
+ '',
182
+ 'Return ONLY valid JSON with this exact shape:',
183
+ '{"isCorrect": boolean, "confidence": number, "reason": string, "matchedAnswer": string | null}',
184
+ '',
185
+ 'Rules:',
186
+ '- Accept synonyms, short paraphrases, and equivalent wording.',
187
+ '- Reject answers that are contradictory, unrelated, or materially incomplete.',
188
+ '- confidence must be from 0 to 1.',
189
+ '- reason must be under 160 characters.',
190
+ '- matchedAnswer must be the matched canonical/alias answer, or null if incorrect.',
191
+ '',
192
+ `Question: ${flashcard.front}`,
193
+ `Canonical answer: ${flashcard.back}`,
194
+ `Accepted aliases: ${JSON.stringify(acceptedAnswers)}`,
195
+ `Student answer: ${input.userAnswer}`,
196
+ ].join('\n');
197
+ try {
198
+ const response = await ai.llm.complete([{ role: 'user', content: prompt }]);
199
+ const content = response.choices?.[0]?.message?.content ?? '';
200
+ const jsonCandidate = extractFirstJsonObject(content);
201
+ if (!jsonCandidate) {
202
+ throw new Error('No JSON object found in grading response');
203
+ }
204
+ const parsed = JSON.parse(jsonCandidate);
205
+ return typedAnswerGradeSchema.parse(parsed);
206
+ }
207
+ catch (error) {
208
+ throw new TRPCError({
209
+ code: 'INTERNAL_SERVER_ERROR',
210
+ message: 'Failed to grade typed answer. Please retry.',
211
+ cause: error,
212
+ });
213
+ }
214
+ }
215
+ async deleteCard(userId, cardId) {
216
+ const card = await this.db.flashcard.findFirst({
217
+ where: {
218
+ id: cardId,
219
+ artifact: { workspace: workspaceAccessWhere(userId) },
220
+ },
221
+ });
222
+ if (!card)
223
+ throw new TRPCError({ code: 'NOT_FOUND' });
224
+ await this.db.flashcard.delete({ where: { id: cardId } });
225
+ return true;
226
+ }
227
+ async deleteSet(userId, setId) {
228
+ const deleted = await this.db.artifact.deleteMany({
229
+ where: {
230
+ id: setId,
231
+ type: ArtifactType.FLASHCARD_SET,
232
+ workspace: workspaceAccessWhere(userId),
233
+ },
234
+ });
235
+ if (deleted.count === 0)
236
+ throw new TRPCError({ code: 'NOT_FOUND' });
237
+ return true;
238
+ }
239
+ async generateFromPrompt(userId, input) {
240
+ const workspace = await this.db.workspace.findFirst({
241
+ where: { id: input.workspaceId, ownerId: userId },
242
+ });
243
+ if (!workspace)
244
+ throw new TRPCError({ code: 'NOT_FOUND' });
245
+ const primarySet = await this.ensurePrimarySet(userId, input.workspaceId);
246
+ const orderOffset = primarySet.flashcards.reduce((max, card) => Math.max(max, card.order), -1);
247
+ try {
248
+ await PusherService.emitTaskComplete(input.workspaceId, 'flash_card_info', {
249
+ status: 'generating',
250
+ numCards: input.numCards,
251
+ difficulty: input.difficulty,
252
+ });
253
+ await this.db.artifact.update({
254
+ where: { id: primarySet.id },
255
+ data: {
256
+ generating: true,
257
+ generatingMetadata: {
258
+ quantity: input.numCards,
259
+ difficulty: input.difficulty.toLowerCase(),
260
+ },
261
+ ...(input.title ? { title: input.title } : {}),
262
+ },
263
+ });
264
+ const ragContext = await workspaceKbService.retrieveContext(input.workspaceId, input.prompt ?? 'flashcard key concepts facts definitions', 8);
265
+ const content = await ai.backend.generateFlashcardQuestions(input.workspaceId, userId, input.numCards, input.difficulty, input.prompt, { ragContext });
266
+ let createdCards = 0;
267
+ try {
268
+ const parsed = typeof content === 'string' ? JSON.parse(content) : content;
269
+ const flashcardData = Array.isArray(parsed) ? parsed : parsed.flashcards || [];
270
+ for (let i = 0; i < Math.min(flashcardData.length, input.numCards); i++) {
271
+ const card = flashcardData[i];
272
+ const front = card.term || card.front || card.question || card.prompt || `Question ${i + 1}`;
273
+ const back = card.definition || card.back || card.answer || card.solution || `Answer ${i + 1}`;
274
+ await this.db.flashcard.create({
275
+ data: {
276
+ artifactId: primarySet.id,
277
+ front,
278
+ back,
279
+ order: orderOffset + 1 + i,
280
+ tags: input.tags ?? ['ai-generated', input.difficulty],
281
+ },
282
+ });
283
+ createdCards++;
284
+ }
285
+ }
286
+ catch (error) {
287
+ console.error('Failed to parse flashcard JSON or create cards:', error);
288
+ const lines = content.split('\n').filter((line) => line.trim());
289
+ for (let i = 0; i < Math.min(lines.length, input.numCards); i++) {
290
+ const line = lines[i];
291
+ if (line.includes(' - ')) {
292
+ const [front, back] = line.split(' - ');
293
+ await this.db.flashcard.create({
294
+ data: {
295
+ artifactId: primarySet.id,
296
+ front: front.trim(),
297
+ back: back.trim(),
298
+ order: orderOffset + 1 + i,
299
+ tags: input.tags ?? ['ai-generated', input.difficulty],
300
+ },
301
+ });
302
+ createdCards++;
303
+ }
304
+ }
305
+ }
306
+ const artifact = await this.db.artifact.update({
307
+ where: { id: primarySet.id },
308
+ data: { generating: false },
309
+ include: { flashcards: true },
310
+ });
311
+ await PusherService.emitFlashcardComplete(input.workspaceId, artifact);
312
+ await notifyArtifactReady(this.db, {
313
+ userId,
314
+ workspaceId: input.workspaceId,
315
+ artifactId: artifact.id,
316
+ artifactType: ArtifactType.FLASHCARD_SET,
317
+ title: artifact.title,
318
+ }).catch(() => { });
319
+ invalidateUserBillingCache(userId);
320
+ return { artifact, createdCards };
321
+ }
322
+ catch (error) {
323
+ await this.db.artifact
324
+ .update({
325
+ where: { id: primarySet.id },
326
+ data: { generating: false },
327
+ })
328
+ .catch(() => { });
329
+ await PusherService.emitError(input.workspaceId, `Failed to generate flashcards: ${error}`, 'flash_card_generation');
330
+ await notifyArtifactFailed(this.db, {
331
+ userId,
332
+ workspaceId: input.workspaceId,
333
+ artifactType: ArtifactType.FLASHCARD_SET,
334
+ message: error instanceof Error ? error.message : 'Flashcard generation failed.',
335
+ }).catch(() => { });
336
+ throw error;
337
+ }
338
+ }
339
+ }
@@ -0,0 +1,23 @@
1
+ import type { PrismaClient } from '@prisma/client';
2
+ import { BaseService } from '../base.service.js';
3
+ export interface UploadAndAnalyzeInput {
4
+ workspaceId: string;
5
+ files: Array<{
6
+ id: string;
7
+ }>;
8
+ generateStudyGuide: boolean;
9
+ generateFlashcards: boolean;
10
+ generateWorksheet: boolean;
11
+ }
12
+ export declare class MediaAnalysisService extends BaseService {
13
+ constructor(db: PrismaClient);
14
+ private updateAnalysisProgress;
15
+ uploadAndAnalyzeMedia(input: UploadAndAnalyzeInput, userId: string): Promise<{
16
+ filename: string;
17
+ artifacts: {
18
+ studyGuide: any | null;
19
+ flashcards: any | null;
20
+ worksheet: any | null;
21
+ };
22
+ }>;
23
+ }