@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,154 @@
1
+ import { TRPCError } from '@trpc/server';
2
+ import { BaseService } from '../base.service.js';
3
+ import { notifyStudyGuideCommentAdded } from '../notifications/notification.service.js';
4
+ export class AnnotationService extends BaseService {
5
+ constructor(db) {
6
+ super(db);
7
+ }
8
+ listHighlights(artifactVersionId) {
9
+ return this.db.studyGuideHighlight.findMany({
10
+ where: { artifactVersionId },
11
+ include: {
12
+ comments: {
13
+ include: {
14
+ user: { select: { id: true, name: true, profilePicture: true } },
15
+ },
16
+ orderBy: { createdAt: 'asc' },
17
+ },
18
+ user: { select: { id: true, name: true, profilePicture: true } },
19
+ },
20
+ orderBy: { startOffset: 'asc' },
21
+ });
22
+ }
23
+ async createHighlight(userId, input) {
24
+ const version = await this.db.artifactVersion.findUnique({
25
+ where: { id: input.artifactVersionId },
26
+ });
27
+ if (!version) {
28
+ throw new TRPCError({ code: 'NOT_FOUND', message: 'Artifact version not found' });
29
+ }
30
+ return this.db.studyGuideHighlight.create({
31
+ data: {
32
+ artifactVersionId: input.artifactVersionId,
33
+ userId,
34
+ startOffset: input.startOffset,
35
+ endOffset: input.endOffset,
36
+ selectedText: input.selectedText,
37
+ ...(input.color && { color: input.color }),
38
+ },
39
+ include: {
40
+ comments: true,
41
+ user: { select: { id: true, name: true, profilePicture: true } },
42
+ },
43
+ });
44
+ }
45
+ async deleteHighlight(userId, highlightId) {
46
+ const highlight = await this.db.studyGuideHighlight.findUnique({
47
+ where: { id: highlightId },
48
+ });
49
+ if (!highlight) {
50
+ throw new TRPCError({ code: 'NOT_FOUND', message: 'Highlight not found' });
51
+ }
52
+ if (highlight.userId !== userId) {
53
+ throw new TRPCError({
54
+ code: 'FORBIDDEN',
55
+ message: 'You can only delete your own highlights',
56
+ });
57
+ }
58
+ await this.db.studyGuideHighlight.delete({ where: { id: highlightId } });
59
+ return { success: true };
60
+ }
61
+ async addComment(userId, input) {
62
+ const highlight = await this.db.studyGuideHighlight.findUnique({
63
+ where: { id: input.highlightId },
64
+ });
65
+ if (!highlight) {
66
+ throw new TRPCError({ code: 'NOT_FOUND', message: 'Highlight not found' });
67
+ }
68
+ const comment = await this.db.studyGuideComment.create({
69
+ data: {
70
+ highlightId: input.highlightId,
71
+ userId,
72
+ content: input.content,
73
+ },
74
+ include: {
75
+ user: { select: { id: true, name: true, profilePicture: true } },
76
+ },
77
+ });
78
+ const after = await this.db.studyGuideHighlight.findUnique({
79
+ where: { id: input.highlightId },
80
+ include: {
81
+ comments: { select: { userId: true } },
82
+ artifactVersion: {
83
+ include: {
84
+ artifact: {
85
+ select: { id: true, title: true, workspaceId: true, type: true },
86
+ },
87
+ },
88
+ },
89
+ },
90
+ });
91
+ if (after?.artifactVersion?.artifact) {
92
+ const art = after.artifactVersion.artifact;
93
+ const recipientIds = new Set();
94
+ if (after.userId !== userId) {
95
+ recipientIds.add(after.userId);
96
+ }
97
+ for (const c of after.comments) {
98
+ if (c.userId !== userId) {
99
+ recipientIds.add(c.userId);
100
+ }
101
+ }
102
+ const authorName = comment.user.name?.trim() || 'Someone';
103
+ await notifyStudyGuideCommentAdded(this.db, {
104
+ authorUserId: userId,
105
+ authorName,
106
+ content: input.content,
107
+ highlightId: input.highlightId,
108
+ commentId: comment.id,
109
+ workspaceId: art.workspaceId,
110
+ artifactId: art.id,
111
+ artifactTitle: art.title,
112
+ recipientUserIds: [...recipientIds],
113
+ }).catch(() => { });
114
+ }
115
+ return comment;
116
+ }
117
+ async updateComment(userId, input) {
118
+ const comment = await this.db.studyGuideComment.findUnique({
119
+ where: { id: input.commentId },
120
+ });
121
+ if (!comment) {
122
+ throw new TRPCError({ code: 'NOT_FOUND', message: 'Comment not found' });
123
+ }
124
+ if (comment.userId !== userId) {
125
+ throw new TRPCError({
126
+ code: 'FORBIDDEN',
127
+ message: 'You can only edit your own comments',
128
+ });
129
+ }
130
+ return this.db.studyGuideComment.update({
131
+ where: { id: input.commentId },
132
+ data: { content: input.content },
133
+ include: {
134
+ user: { select: { id: true, name: true, profilePicture: true } },
135
+ },
136
+ });
137
+ }
138
+ async deleteComment(userId, commentId) {
139
+ const comment = await this.db.studyGuideComment.findUnique({
140
+ where: { id: commentId },
141
+ });
142
+ if (!comment) {
143
+ throw new TRPCError({ code: 'NOT_FOUND', message: 'Comment not found' });
144
+ }
145
+ if (comment.userId !== userId) {
146
+ throw new TRPCError({
147
+ code: 'FORBIDDEN',
148
+ message: 'You can only delete your own comments',
149
+ });
150
+ }
151
+ await this.db.studyGuideComment.delete({ where: { id: commentId } });
152
+ return { success: true };
153
+ }
154
+ }
@@ -0,0 +1,38 @@
1
+ import type { PrismaClient } from '@prisma/client';
2
+ import { BaseService } from '../base.service.js';
3
+ export declare class ArtifactVersionService extends BaseService {
4
+ constructor(db: PrismaClient);
5
+ private assertArtifactAccess;
6
+ private assertVersionAccess;
7
+ listVersions(userId: string, artifactId: string): Promise<{
8
+ id: string;
9
+ version: number;
10
+ createdAt: Date;
11
+ isCurrent: boolean;
12
+ label: string | null;
13
+ createdBy: {
14
+ id: string;
15
+ name: string | null;
16
+ } | null;
17
+ }[]>;
18
+ getVersion(userId: string, artifactVersionId: string): Promise<{
19
+ id: string;
20
+ artifactId: string;
21
+ version: number;
22
+ content: string;
23
+ data: import("@prisma/client/runtime/client").JsonValue;
24
+ createdAt: Date;
25
+ isCurrent: boolean;
26
+ createdBy: string | null;
27
+ }>;
28
+ restoreVersion(userId: string, artifactVersionId: string): Promise<{
29
+ id: string;
30
+ createdAt: Date;
31
+ data: import("@prisma/client/runtime/client").JsonValue | null;
32
+ createdById: string | null;
33
+ artifactId: string;
34
+ content: string;
35
+ version: number;
36
+ }>;
37
+ private versionLabel;
38
+ }
@@ -0,0 +1,129 @@
1
+ import { TRPCError } from '@trpc/server';
2
+ import { BaseService } from '../base.service.js';
3
+ import { workspaceAccessWhere } from '../../repositories/workspace.repository.js';
4
+ export class ArtifactVersionService extends BaseService {
5
+ constructor(db) {
6
+ super(db);
7
+ }
8
+ async assertArtifactAccess(userId, artifactId) {
9
+ const artifact = await this.db.artifact.findFirst({
10
+ where: {
11
+ id: artifactId,
12
+ workspace: workspaceAccessWhere(userId),
13
+ },
14
+ select: { id: true, workspaceId: true },
15
+ });
16
+ if (!artifact) {
17
+ throw new TRPCError({
18
+ code: 'NOT_FOUND',
19
+ message: 'Artifact not found or access denied',
20
+ });
21
+ }
22
+ return artifact;
23
+ }
24
+ async assertVersionAccess(userId, artifactVersionId) {
25
+ const version = await this.db.artifactVersion.findUnique({
26
+ where: { id: artifactVersionId },
27
+ include: {
28
+ artifact: {
29
+ select: { id: true, workspaceId: true, workspace: { select: { id: true } } },
30
+ },
31
+ },
32
+ });
33
+ if (!version) {
34
+ throw new TRPCError({ code: 'NOT_FOUND', message: 'Version not found' });
35
+ }
36
+ const hasAccess = await this.db.workspace.findFirst({
37
+ where: {
38
+ id: version.artifact.workspaceId,
39
+ ...workspaceAccessWhere(userId),
40
+ },
41
+ select: { id: true },
42
+ });
43
+ if (!hasAccess) {
44
+ throw new TRPCError({ code: 'FORBIDDEN', message: 'Access denied' });
45
+ }
46
+ return version;
47
+ }
48
+ async listVersions(userId, artifactId) {
49
+ await this.assertArtifactAccess(userId, artifactId);
50
+ const versions = await this.db.artifactVersion.findMany({
51
+ where: { artifactId },
52
+ orderBy: { version: 'desc' },
53
+ select: {
54
+ id: true,
55
+ version: true,
56
+ createdAt: true,
57
+ data: true,
58
+ createdBy: {
59
+ select: { id: true, name: true },
60
+ },
61
+ },
62
+ });
63
+ const latestId = versions[0]?.id ?? null;
64
+ return versions.map((v) => ({
65
+ id: v.id,
66
+ version: v.version,
67
+ createdAt: v.createdAt,
68
+ isCurrent: v.id === latestId,
69
+ label: this.versionLabel(v.data),
70
+ createdBy: v.createdBy,
71
+ }));
72
+ }
73
+ async getVersion(userId, artifactVersionId) {
74
+ const version = await this.assertVersionAccess(userId, artifactVersionId);
75
+ const latest = await this.db.artifactVersion.findFirst({
76
+ where: { artifactId: version.artifactId },
77
+ orderBy: { version: 'desc' },
78
+ select: { id: true },
79
+ });
80
+ return {
81
+ id: version.id,
82
+ artifactId: version.artifactId,
83
+ version: version.version,
84
+ content: version.content,
85
+ data: version.data,
86
+ createdAt: version.createdAt,
87
+ isCurrent: version.id === latest?.id,
88
+ createdBy: version.createdById,
89
+ };
90
+ }
91
+ async restoreVersion(userId, artifactVersionId) {
92
+ const source = await this.assertVersionAccess(userId, artifactVersionId);
93
+ const latest = await this.db.artifactVersion.findFirst({
94
+ where: { artifactId: source.artifactId },
95
+ orderBy: { version: 'desc' },
96
+ });
97
+ if (latest?.id === source.id) {
98
+ throw new TRPCError({
99
+ code: 'BAD_REQUEST',
100
+ message: 'This is already the current version',
101
+ });
102
+ }
103
+ const nextVersion = (latest?.version ?? 0) + 1;
104
+ const restored = await this.db.artifactVersion.create({
105
+ data: {
106
+ artifactId: source.artifactId,
107
+ version: nextVersion,
108
+ content: source.content,
109
+ data: source.data ?? undefined,
110
+ createdById: userId,
111
+ },
112
+ });
113
+ await this.db.artifact.update({
114
+ where: { id: source.artifactId },
115
+ data: { updatedAt: new Date() },
116
+ });
117
+ return restored;
118
+ }
119
+ versionLabel(data) {
120
+ if (!data || typeof data !== 'object')
121
+ return null;
122
+ const record = data;
123
+ if (typeof record.title === 'string')
124
+ return record.title;
125
+ if (typeof record.description === 'string')
126
+ return record.description;
127
+ return null;
128
+ }
129
+ }
@@ -0,0 +1,127 @@
1
+ import type { PrismaClient } from '@prisma/client';
2
+ import { BaseService } from '../base.service.js';
3
+ /**
4
+ * NOTE: These procedures currently do not verify that the caller belongs to
5
+ * the target workspace or channel. That's pre-existing behavior and is
6
+ * listed in REFACTOR_NOTES.md as a follow-up.
7
+ */
8
+ export declare class ChatService extends BaseService {
9
+ constructor(db: PrismaClient);
10
+ getChannels(workspaceId: string): Promise<{
11
+ id: string;
12
+ name: string;
13
+ createdAt: Date;
14
+ workspaceId: string;
15
+ }[]>;
16
+ getChannel(input: {
17
+ workspaceId?: string;
18
+ channelId?: string;
19
+ }): Promise<{
20
+ chats: ({
21
+ user: {
22
+ id: string;
23
+ name: string | null;
24
+ } | null;
25
+ } & {
26
+ userId: string | null;
27
+ id: string;
28
+ createdAt: Date;
29
+ updatedAt: Date;
30
+ message: string;
31
+ channelId: string;
32
+ })[];
33
+ } & {
34
+ id: string;
35
+ name: string;
36
+ createdAt: Date;
37
+ workspaceId: string;
38
+ }>;
39
+ removeChannel(input: {
40
+ workspaceId: string;
41
+ channelId: string;
42
+ }): Promise<{
43
+ success: boolean;
44
+ }>;
45
+ editChannel(input: {
46
+ workspaceId: string;
47
+ channelId: string;
48
+ name: string;
49
+ }): Promise<{
50
+ chats: ({
51
+ user: {
52
+ id: string;
53
+ name: string | null;
54
+ } | null;
55
+ } & {
56
+ userId: string | null;
57
+ id: string;
58
+ createdAt: Date;
59
+ updatedAt: Date;
60
+ message: string;
61
+ channelId: string;
62
+ })[];
63
+ } & {
64
+ id: string;
65
+ name: string;
66
+ createdAt: Date;
67
+ workspaceId: string;
68
+ }>;
69
+ createChannel(input: {
70
+ workspaceId: string;
71
+ name: string;
72
+ }): Promise<{
73
+ chats: ({
74
+ user: {
75
+ id: string;
76
+ name: string | null;
77
+ } | null;
78
+ } & {
79
+ userId: string | null;
80
+ id: string;
81
+ createdAt: Date;
82
+ updatedAt: Date;
83
+ message: string;
84
+ channelId: string;
85
+ })[];
86
+ } & {
87
+ id: string;
88
+ name: string;
89
+ createdAt: Date;
90
+ workspaceId: string;
91
+ }>;
92
+ postMessage(userId: string, input: {
93
+ channelId: string;
94
+ message: string;
95
+ }): Promise<{
96
+ user: {
97
+ id: string;
98
+ name: string | null;
99
+ } | null;
100
+ } & {
101
+ userId: string | null;
102
+ id: string;
103
+ createdAt: Date;
104
+ updatedAt: Date;
105
+ message: string;
106
+ channelId: string;
107
+ }>;
108
+ editMessage(userId: string, input: {
109
+ chatId: string;
110
+ message: string;
111
+ }): Promise<{
112
+ user: {
113
+ id: string;
114
+ name: string | null;
115
+ } | null;
116
+ } & {
117
+ userId: string | null;
118
+ id: string;
119
+ createdAt: Date;
120
+ updatedAt: Date;
121
+ message: string;
122
+ channelId: string;
123
+ }>;
124
+ deleteMessage(userId: string, chatId: string): Promise<{
125
+ success: boolean;
126
+ }>;
127
+ }
@@ -0,0 +1,182 @@
1
+ import { TRPCError } from '@trpc/server';
2
+ import { BaseService } from '../base.service.js';
3
+ import PusherService from '../../lib/pusher.js';
4
+ /**
5
+ * NOTE: These procedures currently do not verify that the caller belongs to
6
+ * the target workspace or channel. That's pre-existing behavior and is
7
+ * listed in REFACTOR_NOTES.md as a follow-up.
8
+ */
9
+ export class ChatService extends BaseService {
10
+ constructor(db) {
11
+ super(db);
12
+ }
13
+ async getChannels(workspaceId) {
14
+ const channels = await this.db.channel.findMany({
15
+ where: { workspaceId },
16
+ include: {
17
+ chats: {
18
+ include: {
19
+ user: {
20
+ select: {
21
+ id: true,
22
+ name: true,
23
+ profilePicture: {
24
+ select: { id: true, name: true, url: true },
25
+ },
26
+ },
27
+ },
28
+ },
29
+ },
30
+ },
31
+ });
32
+ if (!channels) {
33
+ const defaultChannel = await this.db.channel.create({
34
+ data: { workspaceId, name: 'General' },
35
+ });
36
+ return [defaultChannel];
37
+ }
38
+ return channels;
39
+ }
40
+ async getChannel(input) {
41
+ if (!input.channelId && input.workspaceId) {
42
+ const defaultChannel = await this.db.channel.create({
43
+ data: { workspaceId: input.workspaceId, name: 'General' },
44
+ include: {
45
+ chats: {
46
+ include: {
47
+ user: { select: { id: true, name: true } },
48
+ },
49
+ },
50
+ },
51
+ });
52
+ await PusherService.emitTaskComplete(input.workspaceId, 'new_channel', {
53
+ channelId: defaultChannel.id,
54
+ workspaceId: input.workspaceId,
55
+ name: 'General',
56
+ createdAt: defaultChannel.createdAt,
57
+ });
58
+ return defaultChannel;
59
+ }
60
+ const channel = await this.db.channel.findUnique({
61
+ where: { id: input.channelId },
62
+ include: {
63
+ chats: {
64
+ include: {
65
+ user: { select: { id: true, name: true } },
66
+ },
67
+ },
68
+ },
69
+ });
70
+ if (!channel) {
71
+ throw new TRPCError({ code: 'NOT_FOUND', message: 'Channel not found' });
72
+ }
73
+ return channel;
74
+ }
75
+ async removeChannel(input) {
76
+ await this.db.channel.delete({ where: { id: input.channelId } });
77
+ await PusherService.emitTaskComplete(input.workspaceId, 'remove_channel', {
78
+ channelId: input.channelId,
79
+ deletedAt: new Date().toISOString(),
80
+ });
81
+ return { success: true };
82
+ }
83
+ async editChannel(input) {
84
+ const channel = await this.db.channel.update({
85
+ where: { id: input.channelId },
86
+ data: { name: input.name },
87
+ include: {
88
+ chats: {
89
+ include: {
90
+ user: { select: { id: true, name: true } },
91
+ },
92
+ },
93
+ },
94
+ });
95
+ await PusherService.emitTaskComplete(input.workspaceId, 'edit_channel', {
96
+ channelId: input.channelId,
97
+ workspaceId: input.workspaceId,
98
+ name: input.name,
99
+ });
100
+ return channel;
101
+ }
102
+ async createChannel(input) {
103
+ const channel = await this.db.channel.create({
104
+ data: { workspaceId: input.workspaceId, name: input.name },
105
+ include: {
106
+ chats: {
107
+ include: {
108
+ user: { select: { id: true, name: true } },
109
+ },
110
+ },
111
+ },
112
+ });
113
+ await PusherService.emitTaskComplete(input.workspaceId, 'new_channel', {
114
+ channelId: channel.id,
115
+ workspaceId: input.workspaceId,
116
+ name: input.name,
117
+ createdAt: channel.createdAt,
118
+ });
119
+ return channel;
120
+ }
121
+ async postMessage(userId, input) {
122
+ const channel = await this.db.channel.findUnique({
123
+ where: { id: input.channelId },
124
+ include: {
125
+ chats: {
126
+ include: {
127
+ user: { select: { id: true, name: true } },
128
+ },
129
+ },
130
+ },
131
+ });
132
+ if (!channel) {
133
+ throw new TRPCError({ code: 'NOT_FOUND', message: 'Channel not found' });
134
+ }
135
+ const chat = await this.db.chat.create({
136
+ data: {
137
+ channelId: input.channelId,
138
+ userId,
139
+ message: input.message,
140
+ },
141
+ include: {
142
+ user: { select: { id: true, name: true } },
143
+ },
144
+ });
145
+ await PusherService.emitChannelEvent(input.channelId, 'new_message', chat);
146
+ return chat;
147
+ }
148
+ async editMessage(userId, input) {
149
+ const chat = await this.db.chat.findUnique({
150
+ where: { id: input.chatId },
151
+ include: { user: { select: { id: true, name: true } } },
152
+ });
153
+ if (!chat) {
154
+ throw new TRPCError({ code: 'NOT_FOUND', message: 'Chat message not found' });
155
+ }
156
+ if (chat.userId !== userId) {
157
+ throw new TRPCError({ code: 'FORBIDDEN', message: 'Not your message to edit' });
158
+ }
159
+ const updatedChat = await this.db.chat.update({
160
+ where: { id: input.chatId },
161
+ data: { message: input.message },
162
+ include: { user: { select: { id: true, name: true } } },
163
+ });
164
+ await PusherService.emitChannelEvent(chat.channelId, 'edit_message', updatedChat);
165
+ return updatedChat;
166
+ }
167
+ async deleteMessage(userId, chatId) {
168
+ const chat = await this.db.chat.findUnique({
169
+ where: { id: chatId },
170
+ include: { user: { select: { id: true, name: true } } },
171
+ });
172
+ if (!chat) {
173
+ throw new TRPCError({ code: 'NOT_FOUND', message: 'Chat message not found' });
174
+ }
175
+ if (chat.userId !== userId) {
176
+ throw new TRPCError({ code: 'FORBIDDEN', message: 'Not your message to delete' });
177
+ }
178
+ await this.db.chat.delete({ where: { id: chatId } });
179
+ await PusherService.emitChannelEvent(chat.channelId, 'delete_message', chat);
180
+ return { success: true };
181
+ }
182
+ }
@@ -0,0 +1,18 @@
1
+ import type { PrismaClient } from '@prisma/client';
2
+ import { BaseService } from '../base.service.js';
3
+ export declare class StudyGuideService extends BaseService {
4
+ constructor(db: PrismaClient);
5
+ get(userId: string, workspaceId: string): Promise<{
6
+ artifactId: string;
7
+ title: string;
8
+ latestVersion: {
9
+ id: string;
10
+ createdAt: Date;
11
+ data: import("@prisma/client/runtime/client").JsonValue | null;
12
+ createdById: string | null;
13
+ artifactId: string;
14
+ content: string;
15
+ version: number;
16
+ };
17
+ }>;
18
+ }