@studious-lms/server 1.1.26 → 1.2.6

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 (486) hide show
  1. package/.coderabbit.yaml +9 -0
  2. package/.env.example +53 -0
  3. package/.env.test.example +37 -0
  4. package/README.md +34 -7
  5. package/dist/exportType.d.ts.map +1 -1
  6. package/dist/exportType.js +4 -0
  7. package/dist/exportType.js.map +1 -0
  8. package/dist/index.d.ts +1 -1
  9. package/dist/index.d.ts.map +1 -1
  10. package/dist/index.js +212 -51
  11. package/dist/index.js.map +1 -0
  12. package/dist/instrument.d.ts +2 -0
  13. package/dist/instrument.d.ts.map +1 -0
  14. package/dist/instrument.js +18 -0
  15. package/dist/instrument.js.map +1 -0
  16. package/dist/lib/config/env.d.ts +190 -0
  17. package/dist/lib/config/env.d.ts.map +1 -0
  18. package/dist/lib/config/env.js +121 -0
  19. package/dist/lib/config/env.js.map +1 -0
  20. package/dist/lib/fileUpload.d.ts +2 -2
  21. package/dist/lib/fileUpload.d.ts.map +1 -1
  22. package/dist/lib/fileUpload.js +15 -5
  23. package/dist/lib/fileUpload.js.map +1 -0
  24. package/dist/lib/googleCloudStorage.d.ts +6 -0
  25. package/dist/lib/googleCloudStorage.d.ts.map +1 -1
  26. package/dist/lib/googleCloudStorage.js +26 -6
  27. package/dist/lib/googleCloudStorage.js.map +1 -0
  28. package/dist/lib/jsonConversion.d.ts.map +1 -1
  29. package/dist/lib/jsonConversion.js +16 -14
  30. package/dist/lib/jsonConversion.js.map +1 -0
  31. package/dist/lib/jsonStyles.d.ts.map +1 -1
  32. package/dist/lib/jsonStyles.js +4 -0
  33. package/dist/lib/jsonStyles.js.map +1 -0
  34. package/dist/lib/notificationHandler.d.ts +2 -2
  35. package/dist/lib/notificationHandler.d.ts.map +1 -1
  36. package/dist/lib/notificationHandler.js +4 -0
  37. package/dist/lib/notificationHandler.js.map +1 -0
  38. package/dist/lib/prisma.d.ts +2 -2
  39. package/dist/lib/prisma.d.ts.map +1 -1
  40. package/dist/lib/prisma.js +24 -1
  41. package/dist/lib/prisma.js.map +1 -0
  42. package/dist/lib/pusher.d.ts +4 -1
  43. package/dist/lib/pusher.d.ts.map +1 -1
  44. package/dist/lib/pusher.js +14 -6
  45. package/dist/lib/pusher.js.map +1 -0
  46. package/dist/lib/redis.d.ts +5 -0
  47. package/dist/lib/redis.d.ts.map +1 -0
  48. package/dist/lib/redis.js +53 -0
  49. package/dist/lib/redis.js.map +1 -0
  50. package/dist/lib/thumbnailGenerator.d.ts +0 -21
  51. package/dist/lib/thumbnailGenerator.d.ts.map +1 -1
  52. package/dist/lib/thumbnailGenerator.js +159 -158
  53. package/dist/lib/thumbnailGenerator.js.map +1 -0
  54. package/dist/middleware/auth.d.ts.map +1 -1
  55. package/dist/middleware/auth.js +41 -93
  56. package/dist/middleware/auth.js.map +1 -0
  57. package/dist/middleware/logging.d.ts.map +1 -1
  58. package/dist/middleware/logging.js +4 -0
  59. package/dist/middleware/logging.js.map +1 -0
  60. package/dist/middleware/security.d.ts +5 -0
  61. package/dist/middleware/security.d.ts.map +1 -0
  62. package/dist/middleware/security.js +77 -0
  63. package/dist/middleware/security.js.map +1 -0
  64. package/dist/models/agenda.d.ts +97 -0
  65. package/dist/models/agenda.d.ts.map +1 -0
  66. package/dist/models/agenda.js +40 -0
  67. package/dist/models/agenda.js.map +1 -0
  68. package/dist/models/announcement.d.ts +223 -0
  69. package/dist/models/announcement.d.ts.map +1 -0
  70. package/dist/models/announcement.js +120 -0
  71. package/dist/models/announcement.js.map +1 -0
  72. package/dist/models/assignment.d.ts +1292 -0
  73. package/dist/models/assignment.d.ts.map +1 -0
  74. package/dist/models/assignment.js +309 -0
  75. package/dist/models/assignment.js.map +1 -0
  76. package/dist/models/attendance.d.ts +180 -0
  77. package/dist/models/attendance.d.ts.map +1 -0
  78. package/dist/models/attendance.js +188 -0
  79. package/dist/models/attendance.js.map +1 -0
  80. package/dist/models/auth.d.ts +153 -0
  81. package/dist/models/auth.d.ts.map +1 -0
  82. package/dist/models/auth.js +217 -0
  83. package/dist/models/auth.js.map +1 -0
  84. package/dist/models/class.d.ts +439 -0
  85. package/dist/models/class.d.ts.map +1 -0
  86. package/dist/models/class.js +546 -0
  87. package/dist/models/class.js.map +1 -0
  88. package/dist/models/comment.d.ts +171 -0
  89. package/dist/models/comment.d.ts.map +1 -0
  90. package/dist/models/comment.js +138 -0
  91. package/dist/models/comment.js.map +1 -0
  92. package/dist/models/conversation.d.ts +164 -0
  93. package/dist/models/conversation.d.ts.map +1 -0
  94. package/dist/models/conversation.js +175 -0
  95. package/dist/models/conversation.js.map +1 -0
  96. package/dist/models/event.d.ts +295 -0
  97. package/dist/models/event.d.ts.map +1 -0
  98. package/dist/models/event.js +145 -0
  99. package/dist/models/event.js.map +1 -0
  100. package/dist/models/file.d.ts +536 -0
  101. package/dist/models/file.d.ts.map +1 -0
  102. package/dist/models/file.js +126 -0
  103. package/dist/models/file.js.map +1 -0
  104. package/dist/models/folder.d.ts +295 -0
  105. package/dist/models/folder.d.ts.map +1 -0
  106. package/dist/models/folder.js +202 -0
  107. package/dist/models/folder.js.map +1 -0
  108. package/dist/models/labChat.d.ts +243 -0
  109. package/dist/models/labChat.d.ts.map +1 -0
  110. package/dist/models/labChat.js +204 -0
  111. package/dist/models/labChat.js.map +1 -0
  112. package/dist/models/marketing.d.ts +72 -0
  113. package/dist/models/marketing.d.ts.map +1 -0
  114. package/dist/models/marketing.js +26 -0
  115. package/dist/models/marketing.js.map +1 -0
  116. package/dist/models/message.d.ts +100 -0
  117. package/dist/models/message.d.ts.map +1 -0
  118. package/dist/models/message.js +131 -0
  119. package/dist/models/message.js.map +1 -0
  120. package/dist/models/newtonChat.d.ts +72 -0
  121. package/dist/models/newtonChat.d.ts.map +1 -0
  122. package/dist/models/newtonChat.js +61 -0
  123. package/dist/models/newtonChat.js.map +1 -0
  124. package/dist/models/notification.d.ts +65 -0
  125. package/dist/models/notification.d.ts.map +1 -0
  126. package/dist/models/notification.js +46 -0
  127. package/dist/models/notification.js.map +1 -0
  128. package/dist/models/section.d.ts +102 -0
  129. package/dist/models/section.d.ts.map +1 -0
  130. package/dist/models/section.js +83 -0
  131. package/dist/models/section.js.map +1 -0
  132. package/dist/models/user.d.ts +39 -0
  133. package/dist/models/user.d.ts.map +1 -0
  134. package/dist/models/user.js +38 -0
  135. package/dist/models/user.js.map +1 -0
  136. package/dist/models/worksheet.d.ts +460 -0
  137. package/dist/models/worksheet.d.ts.map +1 -0
  138. package/dist/models/worksheet.js +200 -0
  139. package/dist/models/worksheet.js.map +1 -0
  140. package/dist/pipelines/aiLabChat.d.ts +21 -0
  141. package/dist/pipelines/aiLabChat.d.ts.map +1 -0
  142. package/dist/pipelines/aiLabChat.js +460 -0
  143. package/dist/pipelines/aiLabChat.js.map +1 -0
  144. package/dist/pipelines/aiNewtonChat.d.ts +30 -0
  145. package/dist/pipelines/aiNewtonChat.d.ts.map +1 -0
  146. package/dist/pipelines/aiNewtonChat.js +289 -0
  147. package/dist/pipelines/aiNewtonChat.js.map +1 -0
  148. package/dist/pipelines/gradeWorksheet.d.ts +30 -0
  149. package/dist/pipelines/gradeWorksheet.d.ts.map +1 -0
  150. package/dist/pipelines/gradeWorksheet.js +252 -0
  151. package/dist/pipelines/gradeWorksheet.js.map +1 -0
  152. package/dist/routers/_app.d.ts +6438 -3910
  153. package/dist/routers/_app.d.ts.map +1 -1
  154. package/dist/routers/_app.js +10 -0
  155. package/dist/routers/_app.js.map +1 -0
  156. package/dist/routers/agenda.d.ts +58 -6
  157. package/dist/routers/agenda.d.ts.map +1 -1
  158. package/dist/routers/agenda.js +6 -58
  159. package/dist/routers/agenda.js.map +1 -0
  160. package/dist/routers/announcement.d.ts +325 -6
  161. package/dist/routers/announcement.d.ts.map +1 -1
  162. package/dist/routers/announcement.js +543 -77
  163. package/dist/routers/announcement.js.map +1 -0
  164. package/dist/routers/assignment.d.ts +419 -357
  165. package/dist/routers/assignment.d.ts.map +1 -1
  166. package/dist/routers/assignment.js +100 -1689
  167. package/dist/routers/assignment.js.map +1 -0
  168. package/dist/routers/attendance.d.ts +20 -9
  169. package/dist/routers/attendance.d.ts.map +1 -1
  170. package/dist/routers/attendance.js +10 -263
  171. package/dist/routers/attendance.js.map +1 -0
  172. package/dist/routers/auth.d.ts +21 -1
  173. package/dist/routers/auth.d.ts.map +1 -1
  174. package/dist/routers/auth.js +37 -241
  175. package/dist/routers/auth.js.map +1 -0
  176. package/dist/routers/class.d.ts +198 -68
  177. package/dist/routers/class.d.ts.map +1 -1
  178. package/dist/routers/class.js +88 -909
  179. package/dist/routers/class.js.map +1 -0
  180. package/dist/routers/comment.d.ts +153 -0
  181. package/dist/routers/comment.d.ts.map +1 -0
  182. package/dist/routers/comment.js +58 -0
  183. package/dist/routers/comment.js.map +1 -0
  184. package/dist/routers/conversation.d.ts +73 -3
  185. package/dist/routers/conversation.d.ts.map +1 -1
  186. package/dist/routers/conversation.js +23 -265
  187. package/dist/routers/conversation.js.map +1 -0
  188. package/dist/routers/event.d.ts +46 -37
  189. package/dist/routers/event.d.ts.map +1 -1
  190. package/dist/routers/event.js +15 -431
  191. package/dist/routers/event.js.map +1 -0
  192. package/dist/routers/file.d.ts +4 -2
  193. package/dist/routers/file.d.ts.map +1 -1
  194. package/dist/routers/file.js +11 -298
  195. package/dist/routers/file.js.map +1 -0
  196. package/dist/routers/folder.d.ts +21 -14
  197. package/dist/routers/folder.d.ts.map +1 -1
  198. package/dist/routers/folder.js +36 -743
  199. package/dist/routers/folder.js.map +1 -0
  200. package/dist/routers/labChat.d.ts +12 -9
  201. package/dist/routers/labChat.d.ts.map +1 -1
  202. package/dist/routers/labChat.js +21 -885
  203. package/dist/routers/labChat.js.map +1 -0
  204. package/dist/routers/marketing.d.ts +2 -2
  205. package/dist/routers/marketing.d.ts.map +1 -1
  206. package/dist/routers/marketing.js +9 -54
  207. package/dist/routers/marketing.js.map +1 -0
  208. package/dist/routers/message.d.ts +2 -1
  209. package/dist/routers/message.d.ts.map +1 -1
  210. package/dist/routers/message.js +29 -519
  211. package/dist/routers/message.js.map +1 -0
  212. package/dist/routers/newtonChat.d.ts +55 -0
  213. package/dist/routers/newtonChat.d.ts.map +1 -0
  214. package/dist/routers/newtonChat.js +22 -0
  215. package/dist/routers/newtonChat.js.map +1 -0
  216. package/dist/routers/notifications.d.ts +8 -8
  217. package/dist/routers/notifications.d.ts.map +1 -1
  218. package/dist/routers/notifications.js +20 -81
  219. package/dist/routers/notifications.js.map +1 -0
  220. package/dist/routers/section.d.ts +23 -8
  221. package/dist/routers/section.d.ts.map +1 -1
  222. package/dist/routers/section.js +23 -273
  223. package/dist/routers/section.js.map +1 -0
  224. package/dist/routers/user.d.ts +1 -1
  225. package/dist/routers/user.d.ts.map +1 -1
  226. package/dist/routers/user.js +34 -204
  227. package/dist/routers/user.js.map +1 -0
  228. package/dist/routers/worksheet.d.ts +362 -0
  229. package/dist/routers/worksheet.d.ts.map +1 -0
  230. package/dist/routers/worksheet.js +153 -0
  231. package/dist/routers/worksheet.js.map +1 -0
  232. package/dist/seedDatabase.d.ts +2 -3
  233. package/dist/seedDatabase.d.ts.map +1 -1
  234. package/dist/seedDatabase.js +309 -288
  235. package/dist/seedDatabase.js.map +1 -0
  236. package/dist/server/pipelines/aiLabChat.d.ts +21 -0
  237. package/dist/server/pipelines/aiLabChat.d.ts.map +1 -0
  238. package/dist/server/pipelines/aiLabChat.js +456 -0
  239. package/dist/server/pipelines/aiLabChat.js.map +1 -0
  240. package/dist/server/pipelines/aiNewtonChat.d.ts +30 -0
  241. package/dist/server/pipelines/aiNewtonChat.d.ts.map +1 -0
  242. package/dist/server/pipelines/aiNewtonChat.js +285 -0
  243. package/dist/server/pipelines/aiNewtonChat.js.map +1 -0
  244. package/dist/server/pipelines/gradeWorksheet.d.ts +30 -0
  245. package/dist/server/pipelines/gradeWorksheet.d.ts.map +1 -0
  246. package/dist/server/pipelines/gradeWorksheet.js +248 -0
  247. package/dist/server/pipelines/gradeWorksheet.js.map +1 -0
  248. package/dist/services/agenda.d.ts +100 -0
  249. package/dist/services/agenda.d.ts.map +1 -0
  250. package/dist/services/agenda.js +21 -0
  251. package/dist/services/agenda.js.map +1 -0
  252. package/dist/services/announcement.d.ts +135 -0
  253. package/dist/services/announcement.d.ts.map +1 -0
  254. package/dist/services/announcement.js +223 -0
  255. package/dist/services/announcement.js.map +1 -0
  256. package/dist/services/assignment.d.ts +1462 -0
  257. package/dist/services/assignment.d.ts.map +1 -0
  258. package/dist/services/assignment.js +898 -0
  259. package/dist/services/assignment.js.map +1 -0
  260. package/dist/services/attendance.d.ts +93 -0
  261. package/dist/services/attendance.d.ts.map +1 -0
  262. package/dist/services/attendance.js +61 -0
  263. package/dist/services/attendance.js.map +1 -0
  264. package/dist/services/auth.d.ts +68 -0
  265. package/dist/services/auth.d.ts.map +1 -0
  266. package/dist/services/auth.js +218 -0
  267. package/dist/services/auth.js.map +1 -0
  268. package/dist/services/class.d.ts +621 -0
  269. package/dist/services/class.d.ts.map +1 -0
  270. package/dist/services/class.js +474 -0
  271. package/dist/services/class.js.map +1 -0
  272. package/dist/services/comment.d.ts +100 -0
  273. package/dist/services/comment.d.ts.map +1 -0
  274. package/dist/services/comment.js +83 -0
  275. package/dist/services/comment.js.map +1 -0
  276. package/dist/services/conversation.d.ts +159 -0
  277. package/dist/services/conversation.d.ts.map +1 -0
  278. package/dist/services/conversation.js +138 -0
  279. package/dist/services/conversation.js.map +1 -0
  280. package/dist/services/event.d.ts +216 -0
  281. package/dist/services/event.d.ts.map +1 -0
  282. package/dist/services/event.js +168 -0
  283. package/dist/services/event.js.map +1 -0
  284. package/dist/services/file.d.ts +74 -0
  285. package/dist/services/file.d.ts.map +1 -0
  286. package/dist/services/file.js +133 -0
  287. package/dist/services/file.js.map +1 -0
  288. package/dist/services/folder.d.ts +239 -0
  289. package/dist/services/folder.d.ts.map +1 -0
  290. package/dist/services/folder.js +248 -0
  291. package/dist/services/folder.js.map +1 -0
  292. package/dist/services/labChat.d.ts +165 -0
  293. package/dist/services/labChat.d.ts.map +1 -0
  294. package/dist/services/labChat.js +289 -0
  295. package/dist/services/labChat.js.map +1 -0
  296. package/dist/services/marketing.d.ts +50 -0
  297. package/dist/services/marketing.d.ts.map +1 -0
  298. package/dist/services/marketing.js +32 -0
  299. package/dist/services/marketing.js.map +1 -0
  300. package/dist/services/message.d.ts +95 -0
  301. package/dist/services/message.d.ts.map +1 -0
  302. package/dist/services/message.js +350 -0
  303. package/dist/services/message.js.map +1 -0
  304. package/dist/services/newtonChat.d.ts +22 -0
  305. package/dist/services/newtonChat.d.ts.map +1 -0
  306. package/dist/services/newtonChat.js +174 -0
  307. package/dist/services/newtonChat.js.map +1 -0
  308. package/dist/services/notification.d.ts +65 -0
  309. package/dist/services/notification.d.ts.map +1 -0
  310. package/dist/services/notification.js +33 -0
  311. package/dist/services/notification.js.map +1 -0
  312. package/dist/services/section.d.ts +53 -0
  313. package/dist/services/section.d.ts.map +1 -0
  314. package/dist/services/section.js +199 -0
  315. package/dist/services/section.js.map +1 -0
  316. package/dist/services/user.d.ts +48 -0
  317. package/dist/services/user.d.ts.map +1 -0
  318. package/dist/services/user.js +141 -0
  319. package/dist/services/user.js.map +1 -0
  320. package/dist/services/worksheet.d.ts +239 -0
  321. package/dist/services/worksheet.d.ts.map +1 -0
  322. package/dist/services/worksheet.js +235 -0
  323. package/dist/services/worksheet.js.map +1 -0
  324. package/dist/socket/handlers.d.ts.map +1 -1
  325. package/dist/socket/handlers.js +4 -0
  326. package/dist/socket/handlers.js.map +1 -0
  327. package/dist/trpc.d.ts.map +1 -1
  328. package/dist/trpc.js +4 -0
  329. package/dist/trpc.js.map +1 -0
  330. package/dist/types/trpc.d.ts.map +1 -1
  331. package/dist/types/trpc.js +4 -0
  332. package/dist/types/trpc.js.map +1 -0
  333. package/dist/utils/aiUser.d.ts +1 -3
  334. package/dist/utils/aiUser.d.ts.map +1 -1
  335. package/dist/utils/aiUser.js +8 -3
  336. package/dist/utils/aiUser.js.map +1 -0
  337. package/dist/utils/email.d.ts +12 -1
  338. package/dist/utils/email.d.ts.map +1 -1
  339. package/dist/utils/email.js +26 -4
  340. package/dist/utils/email.js.map +1 -0
  341. package/dist/utils/generateInviteCode.d.ts +1 -2
  342. package/dist/utils/generateInviteCode.d.ts.map +1 -1
  343. package/dist/utils/generateInviteCode.js +5 -2
  344. package/dist/utils/generateInviteCode.js.map +1 -0
  345. package/dist/utils/inference.d.ts +8 -0
  346. package/dist/utils/inference.d.ts.map +1 -1
  347. package/dist/utils/inference.js +78 -10
  348. package/dist/utils/inference.js.map +1 -0
  349. package/dist/utils/logger.d.ts +3 -0
  350. package/dist/utils/logger.d.ts.map +1 -1
  351. package/dist/utils/logger.js +8 -1
  352. package/dist/utils/logger.js.map +1 -0
  353. package/dist/utils/prismaErrorHandler.d.ts.map +1 -1
  354. package/dist/utils/prismaErrorHandler.js +7 -0
  355. package/dist/utils/prismaErrorHandler.js.map +1 -0
  356. package/dist/utils/prismaWrapper.d.ts +1 -0
  357. package/dist/utils/prismaWrapper.d.ts.map +1 -1
  358. package/dist/utils/prismaWrapper.js +8 -0
  359. package/dist/utils/prismaWrapper.js.map +1 -0
  360. package/docker-compose.yml +19 -0
  361. package/package.json +21 -4
  362. package/prisma/migrations/20251109122857_annuoncements_comments/migration.sql +30 -0
  363. package/prisma/migrations/20251109135555_reactions_announcements_comments/migration.sql +35 -0
  364. package/prisma/schema.prisma +180 -12
  365. package/scripts/test-pre-push.ts +14 -0
  366. package/src/index.ts +247 -52
  367. package/src/instrument.ts +15 -0
  368. package/src/lib/config/env.ts +132 -0
  369. package/src/lib/fileUpload.ts +13 -6
  370. package/src/lib/googleCloudStorage.ts +23 -6
  371. package/src/lib/jsonConversion.ts +12 -14
  372. package/src/lib/prisma.ts +23 -2
  373. package/src/lib/pusher.ts +11 -6
  374. package/src/lib/redis.ts +56 -0
  375. package/src/lib/thumbnailGenerator.ts +170 -168
  376. package/src/middleware/auth.ts +86 -137
  377. package/src/middleware/security.ts +80 -0
  378. package/src/models/agenda.ts +46 -0
  379. package/src/models/announcement.ts +134 -0
  380. package/src/models/assignment.ts +322 -0
  381. package/src/models/attendance.ts +208 -0
  382. package/src/models/auth.ts +247 -0
  383. package/src/models/class.ts +598 -0
  384. package/src/models/comment.ts +152 -0
  385. package/src/models/conversation.ts +200 -0
  386. package/src/models/event.ts +177 -0
  387. package/src/models/file.ts +129 -0
  388. package/src/models/folder.ts +225 -0
  389. package/src/models/labChat.ts +213 -0
  390. package/src/models/marketing.ts +45 -0
  391. package/src/models/message.ts +153 -0
  392. package/src/models/newtonChat.ts +70 -0
  393. package/src/models/notification.ts +54 -0
  394. package/src/models/section.ts +98 -0
  395. package/src/models/user.ts +47 -0
  396. package/src/models/worksheet.ts +294 -0
  397. package/src/pipelines/aiLabChat.ts +511 -0
  398. package/src/pipelines/aiNewtonChat.ts +347 -0
  399. package/src/pipelines/gradeWorksheet.ts +286 -0
  400. package/src/routers/_app.ts +6 -0
  401. package/src/routers/agenda.ts +3 -61
  402. package/src/routers/announcement.ts +616 -79
  403. package/src/routers/assignment.ts +148 -1827
  404. package/src/routers/attendance.ts +16 -277
  405. package/src/routers/auth.ts +79 -313
  406. package/src/routers/class.ts +265 -1038
  407. package/src/routers/comment.ts +76 -0
  408. package/src/routers/conversation.ts +53 -284
  409. package/src/routers/event.ts +50 -481
  410. package/src/routers/file.ts +45 -344
  411. package/src/routers/folder.ts +107 -836
  412. package/src/routers/labChat.ts +29 -969
  413. package/src/routers/marketing.ts +35 -77
  414. package/src/routers/message.ts +45 -571
  415. package/src/routers/newtonChat.ts +36 -0
  416. package/src/routers/notifications.ts +32 -82
  417. package/src/routers/section.ts +58 -322
  418. package/src/routers/user.ts +49 -226
  419. package/src/routers/worksheet.ts +252 -0
  420. package/src/seedDatabase.ts +328 -289
  421. package/src/services/agenda.ts +21 -0
  422. package/src/services/announcement.ts +290 -0
  423. package/src/services/assignment.ts +1198 -0
  424. package/src/services/attendance.ts +85 -0
  425. package/src/services/auth.ts +277 -0
  426. package/src/services/class.ts +622 -0
  427. package/src/services/comment.ts +106 -0
  428. package/src/services/conversation.ts +213 -0
  429. package/src/services/event.ts +231 -0
  430. package/src/services/file.ts +167 -0
  431. package/src/services/folder.ts +316 -0
  432. package/src/services/labChat.ts +352 -0
  433. package/src/services/marketing.ts +57 -0
  434. package/src/services/message.ts +461 -0
  435. package/src/services/newtonChat.ts +222 -0
  436. package/src/services/notification.ts +50 -0
  437. package/src/services/section.ts +283 -0
  438. package/src/services/user.ts +172 -0
  439. package/src/services/worksheet.ts +358 -0
  440. package/src/trpc.ts +4 -0
  441. package/src/utils/aiUser.ts +4 -3
  442. package/src/utils/email.ts +33 -4
  443. package/src/utils/generateInviteCode.ts +1 -3
  444. package/src/utils/inference.ts +89 -10
  445. package/src/utils/logger.ts +4 -1
  446. package/src/utils/prismaErrorHandler.ts +3 -0
  447. package/src/utils/prismaWrapper.ts +4 -0
  448. package/tests/globalSetup.ts +62 -0
  449. package/tests/helpers.ts +22 -0
  450. package/tests/middleware/security.test.ts +42 -0
  451. package/tests/routers/agenda.test.ts +138 -0
  452. package/tests/routers/announcement.test.ts +490 -0
  453. package/tests/routers/assignment.test.ts +837 -0
  454. package/tests/routers/attendance.test.ts +160 -0
  455. package/tests/routers/auth.test.ts +171 -0
  456. package/tests/{class.test.ts → routers/class.test.ts} +163 -92
  457. package/tests/routers/comment.test.ts +126 -0
  458. package/tests/routers/conversation.test.ts +145 -0
  459. package/tests/routers/event.test.ts +289 -0
  460. package/tests/routers/folder.test.ts +178 -0
  461. package/tests/routers/labChat.test.ts +115 -0
  462. package/tests/routers/marketing.test.ts +59 -0
  463. package/tests/routers/message.test.ts +123 -0
  464. package/tests/routers/notification.test.ts +69 -0
  465. package/tests/routers/section.test.ts +208 -0
  466. package/tests/server/rateLimit.test.ts +73 -0
  467. package/tests/setup.ts +39 -59
  468. package/tests/user.test.ts +136 -0
  469. package/tests/utils/aiUser.test.ts +22 -0
  470. package/tests/utils/generateInviteCode.test.ts +24 -0
  471. package/tests/utils/logger.test.ts +74 -0
  472. package/tests/utils/prismaErrorHandler.test.ts +101 -0
  473. package/tests/utils/prismaWrapper.test.ts +82 -0
  474. package/tests/worksheet.test.ts +181 -0
  475. package/tsconfig.json +9 -2
  476. package/vitest.config.ts +30 -1
  477. package/vitest.unit.config.ts +21 -0
  478. package/API_SPECIFICATION.md +0 -1597
  479. package/BASE64_REMOVAL_SUMMARY.md +0 -164
  480. package/CHAT_API_SPEC.md +0 -579
  481. package/LAB_CHAT_API_SPEC.md +0 -518
  482. package/dist/routers/school.d.ts +0 -208
  483. package/dist/routers/school.d.ts.map +0 -1
  484. package/dist/routers/school.js +0 -481
  485. package/src/lib/notificationHandler.ts +0 -36
  486. package/tests/auth.test.ts +0 -25
@@ -27,12 +27,12 @@ export declare const notificationRouter: import("@trpc/server").TRPCBuiltRouter<
27
27
  };
28
28
  } & {
29
29
  id: string;
30
+ createdAt: Date;
30
31
  title: string;
31
32
  content: string;
32
- createdAt: Date;
33
+ read: boolean;
33
34
  senderId: string | null;
34
35
  receiverId: string;
35
- read: boolean;
36
36
  })[];
37
37
  meta: object;
38
38
  }>;
@@ -49,12 +49,12 @@ export declare const notificationRouter: import("@trpc/server").TRPCBuiltRouter<
49
49
  };
50
50
  } & {
51
51
  id: string;
52
+ createdAt: Date;
52
53
  title: string;
53
54
  content: string;
54
- createdAt: Date;
55
+ read: boolean;
55
56
  senderId: string | null;
56
57
  receiverId: string;
57
- read: boolean;
58
58
  }) | null;
59
59
  meta: object;
60
60
  }>;
@@ -66,12 +66,12 @@ export declare const notificationRouter: import("@trpc/server").TRPCBuiltRouter<
66
66
  };
67
67
  output: {
68
68
  id: string;
69
+ createdAt: Date;
69
70
  title: string;
70
71
  content: string;
71
- createdAt: Date;
72
+ read: boolean;
72
73
  senderId: string | null;
73
74
  receiverId: string;
74
- read: boolean;
75
75
  };
76
76
  meta: object;
77
77
  }>;
@@ -90,12 +90,12 @@ export declare const notificationRouter: import("@trpc/server").TRPCBuiltRouter<
90
90
  };
91
91
  output: {
92
92
  id: string;
93
+ createdAt: Date;
93
94
  title: string;
94
95
  content: string;
95
- createdAt: Date;
96
+ read: boolean;
96
97
  senderId: string | null;
97
98
  receiverId: string;
98
- read: boolean;
99
99
  };
100
100
  meta: object;
101
101
  }>;
@@ -1 +1 @@
1
- {"version":3,"file":"notifications.d.ts","sourceRoot":"","sources":["../../src/routers/notifications.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AAExB,eAAO,MAAM,kBAAkB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAwF7B,CAAA"}
1
+ {"version":3,"file":"notifications.d.ts","sourceRoot":"/","sources":["routers/notifications.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AASxB,eAAO,MAAM,kBAAkB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAgC7B,CAAC"}
@@ -1,91 +1,30 @@
1
+
2
+ !function(){try{var e="undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof globalThis?globalThis:"undefined"!=typeof self?self:{},n=(new e.Error).stack;n&&(e._sentryDebugIds=e._sentryDebugIds||{},e._sentryDebugIds[n]="2769d494-a91e-58cf-99e2-7fa58208de9a")}catch(e){}}();
1
3
  import { createTRPCRouter, protectedProcedure } from "../trpc.js";
2
- import { prisma } from "../lib/prisma.js";
3
4
  import { z } from "zod";
5
+ import { listNotifications, getNotification, sendNotification, sendToMultiple, markAsRead, } from "../services/notification.js";
4
6
  export const notificationRouter = createTRPCRouter({
5
- list: protectedProcedure.query(async ({ ctx }) => {
6
- const notifications = await prisma.notification.findMany({
7
- where: {
8
- receiverId: ctx.user.id,
9
- },
10
- orderBy: {
11
- createdAt: "desc",
12
- },
13
- include: {
14
- sender: {
15
- select: {
16
- username: true,
17
- },
18
- },
19
- receiver: {
20
- select: {
21
- username: true,
22
- },
23
- },
24
- },
25
- });
26
- return notifications;
27
- }),
28
- get: protectedProcedure.input(z.object({
29
- id: z.string(),
30
- })).query(async ({ ctx, input }) => {
31
- const { id } = input;
32
- const notification = await prisma.notification.findUnique({
33
- where: {
34
- id,
35
- },
36
- include: {
37
- sender: {
38
- select: {
39
- username: true,
40
- },
41
- },
42
- receiver: {
43
- select: {
44
- username: true,
45
- },
46
- },
47
- },
48
- });
49
- return notification;
50
- }),
51
- sendTo: protectedProcedure.input(z.object({
7
+ list: protectedProcedure.query(({ ctx }) => listNotifications(ctx.user.id)),
8
+ get: protectedProcedure
9
+ .input(z.object({ id: z.string() }))
10
+ .query(({ input }) => getNotification(input.id)),
11
+ sendTo: protectedProcedure
12
+ .input(z.object({
52
13
  receiverId: z.string(),
53
14
  title: z.string(),
54
15
  content: z.string(),
55
- })).mutation(async ({ ctx, input }) => {
56
- const { receiverId, title, content } = input;
57
- const notification = await prisma.notification.create({
58
- data: {
59
- receiverId,
60
- title,
61
- content,
62
- },
63
- });
64
- return notification;
65
- }),
66
- sendToMultiple: protectedProcedure.input(z.object({
16
+ }))
17
+ .mutation(({ input }) => sendNotification(input)),
18
+ sendToMultiple: protectedProcedure
19
+ .input(z.object({
67
20
  receiverIds: z.array(z.string()),
68
21
  title: z.string(),
69
22
  content: z.string(),
70
- })).mutation(async ({ ctx, input }) => {
71
- const { receiverIds, title, content } = input;
72
- const notifications = await prisma.notification.createMany({
73
- data: receiverIds.map(receiverId => ({
74
- receiverId,
75
- title,
76
- content,
77
- })),
78
- });
79
- return notifications;
80
- }),
81
- markAsRead: protectedProcedure.input(z.object({
82
- id: z.string(),
83
- })).mutation(async ({ ctx, input }) => {
84
- const { id } = input;
85
- const notification = await prisma.notification.update({
86
- where: { id },
87
- data: { read: true },
88
- });
89
- return notification;
90
- }),
23
+ }))
24
+ .mutation(({ input }) => sendToMultiple(input)),
25
+ markAsRead: protectedProcedure
26
+ .input(z.object({ id: z.string() }))
27
+ .mutation(({ input }) => markAsRead(input.id)),
91
28
  });
29
+ //# sourceMappingURL=notifications.js.map
30
+ //# debugId=2769d494-a91e-58cf-99e2-7fa58208de9a
@@ -0,0 +1 @@
1
+ {"version":3,"file":"notifications.js","sources":["routers/notifications.ts"],"sourceRoot":"/","sourcesContent":["import { createTRPCRouter, protectedProcedure } from \"../trpc.js\";\nimport { z } from \"zod\";\nimport {\n listNotifications,\n getNotification,\n sendNotification,\n sendToMultiple,\n markAsRead,\n} from \"../services/notification.js\";\n\nexport const notificationRouter = createTRPCRouter({\n list: protectedProcedure.query(({ ctx }) =>\n listNotifications(ctx.user!.id)\n ),\n\n get: protectedProcedure\n .input(z.object({ id: z.string() }))\n .query(({ input }) => getNotification(input.id)),\n\n sendTo: protectedProcedure\n .input(\n z.object({\n receiverId: z.string(),\n title: z.string(),\n content: z.string(),\n })\n )\n .mutation(({ input }) => sendNotification(input)),\n\n sendToMultiple: protectedProcedure\n .input(\n z.object({\n receiverIds: z.array(z.string()),\n title: z.string(),\n content: z.string(),\n })\n )\n .mutation(({ input }) => sendToMultiple(input)),\n\n markAsRead: protectedProcedure\n .input(z.object({ id: z.string() }))\n .mutation(({ input }) => markAsRead(input.id)),\n});\n"],"names":[],"mappings":";;AAAA,OAAO,EAAE,gBAAgB,EAAE,kBAAkB,EAAE,MAAM,YAAY,CAAC;AAClE,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AACxB,OAAO,EACL,iBAAiB,EACjB,eAAe,EACf,gBAAgB,EAChB,cAAc,EACd,UAAU,GACX,MAAM,6BAA6B,CAAC;AAErC,MAAM,CAAC,MAAM,kBAAkB,GAAG,gBAAgB,CAAC;IACjD,IAAI,EAAE,kBAAkB,CAAC,KAAK,CAAC,CAAC,EAAE,GAAG,EAAE,EAAE,EAAE,CACzC,iBAAiB,CAAC,GAAG,CAAC,IAAK,CAAC,EAAE,CAAC,CAChC;IAED,GAAG,EAAE,kBAAkB;SACpB,KAAK,CAAC,CAAC,CAAC,MAAM,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC;SACnC,KAAK,CAAC,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC,eAAe,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;IAElD,MAAM,EAAE,kBAAkB;SACvB,KAAK,CACJ,CAAC,CAAC,MAAM,CAAC;QACP,UAAU,EAAE,CAAC,CAAC,MAAM,EAAE;QACtB,KAAK,EAAE,CAAC,CAAC,MAAM,EAAE;QACjB,OAAO,EAAE,CAAC,CAAC,MAAM,EAAE;KACpB,CAAC,CACH;SACA,QAAQ,CAAC,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC;IAEnD,cAAc,EAAE,kBAAkB;SAC/B,KAAK,CACJ,CAAC,CAAC,MAAM,CAAC;QACP,WAAW,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC;QAChC,KAAK,EAAE,CAAC,CAAC,MAAM,EAAE;QACjB,OAAO,EAAE,CAAC,CAAC,MAAM,EAAE;KACpB,CAAC,CACH;SACA,QAAQ,CAAC,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC,cAAc,CAAC,KAAK,CAAC,CAAC;IAEjD,UAAU,EAAE,kBAAkB;SAC3B,KAAK,CAAC,CAAC,CAAC,MAAM,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC;SACnC,QAAQ,CAAC,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;CACjD,CAAC,CAAC","debug_id":"2769d494-a91e-58cf-99e2-7fa58208de9a"}
@@ -16,57 +16,71 @@ export declare const sectionRouter: import("@trpc/server").TRPCBuiltRouter<{
16
16
  };
17
17
  transformer: false;
18
18
  }, import("@trpc/server").TRPCDecorateCreateRouterOptions<{
19
+ exists: import("@trpc/server").TRPCQueryProcedure<{
20
+ input: {
21
+ [x: string]: unknown;
22
+ classId: string;
23
+ id: string;
24
+ };
25
+ output: boolean;
26
+ meta: object;
27
+ }>;
19
28
  create: import("@trpc/server").TRPCMutationProcedure<{
20
29
  input: {
21
- name: string;
30
+ [x: string]: unknown;
22
31
  classId: string;
32
+ name: string;
33
+ id?: string | undefined;
23
34
  color?: string | undefined;
24
35
  };
25
36
  output: {
26
37
  id: string;
38
+ classId: string;
27
39
  name: string;
28
40
  color: string | null;
29
- classId: string;
30
41
  order: number | null;
31
42
  };
32
43
  meta: object;
33
44
  }>;
34
45
  reorder: import("@trpc/server").TRPCMutationProcedure<{
35
46
  input: {
47
+ [x: string]: unknown;
36
48
  classId: string;
37
49
  movedId: string;
38
- position: "start" | "end" | "before" | "after";
50
+ position: "end" | "start" | "before" | "after";
39
51
  targetId?: string | undefined;
40
52
  };
41
53
  output: {
42
54
  id: string;
55
+ classId: string;
43
56
  name: string;
44
57
  color: string | null;
45
- classId: string;
46
58
  order: number | null;
47
59
  } | null;
48
60
  meta: object;
49
61
  }>;
50
62
  update: import("@trpc/server").TRPCMutationProcedure<{
51
63
  input: {
64
+ [x: string]: unknown;
65
+ classId: string;
52
66
  id: string;
53
67
  name: string;
54
- classId: string;
55
68
  color?: string | undefined;
56
69
  };
57
70
  output: {
58
71
  id: string;
72
+ classId: string;
59
73
  name: string;
60
74
  color: string | null;
61
- classId: string;
62
75
  order: number | null;
63
76
  };
64
77
  meta: object;
65
78
  }>;
66
79
  reOrder: import("@trpc/server").TRPCMutationProcedure<{
67
80
  input: {
68
- id: string;
81
+ [x: string]: unknown;
69
82
  classId: string;
83
+ id: string;
70
84
  order: number;
71
85
  };
72
86
  output: {
@@ -76,8 +90,9 @@ export declare const sectionRouter: import("@trpc/server").TRPCBuiltRouter<{
76
90
  }>;
77
91
  delete: import("@trpc/server").TRPCMutationProcedure<{
78
92
  input: {
79
- id: string;
93
+ [x: string]: unknown;
80
94
  classId: string;
95
+ id: string;
81
96
  };
82
97
  output: {
83
98
  id: string;
@@ -1 +1 @@
1
- {"version":3,"file":"section.d.ts","sourceRoot":"","sources":["../../src/routers/section.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AAuBxB,eAAO,MAAM,aAAa;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAkUxB,CAAC"}
1
+ {"version":3,"file":"section.d.ts","sourceRoot":"/","sources":["routers/section.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AAkCxB,eAAO,MAAM,aAAa;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA+CxB,CAAC"}
@@ -1,9 +1,11 @@
1
+
2
+ !function(){try{var e="undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof globalThis?globalThis:"undefined"!=typeof self?self:{},n=(new e.Error).stack;n&&(e._sentryDebugIds=e._sentryDebugIds||{},e._sentryDebugIds[n]="d31ac70e-8c0c-5d80-9d4e-e996c6dc6b08")}catch(e){}}();
1
3
  import { z } from "zod";
2
- import { createTRPCRouter, protectedProcedure } from "../trpc.js";
3
- import { TRPCError } from "@trpc/server";
4
- import { prisma } from "../lib/prisma.js";
4
+ import { createTRPCRouter, protectedClassMemberProcedure, protectedTeacherProcedure, } from "../trpc.js";
5
+ import { sectionExists, createSectionRecord, reorderSection, updateSectionRecord, reOrderSection, deleteSectionRecord, } from "../services/section.js";
5
6
  const createSectionSchema = z.object({
6
7
  classId: z.string(),
8
+ id: z.string().optional(),
7
9
  name: z.string(),
8
10
  color: z.string().optional(),
9
11
  });
@@ -18,285 +20,33 @@ const deleteSectionSchema = z.object({
18
20
  classId: z.string(),
19
21
  });
20
22
  export const sectionRouter = createTRPCRouter({
21
- create: protectedProcedure
23
+ exists: protectedClassMemberProcedure
24
+ .input(z.object({ id: z.string() }))
25
+ .query(({ input }) => sectionExists(input.id)),
26
+ create: protectedTeacherProcedure
22
27
  .input(createSectionSchema)
23
- .mutation(async ({ ctx, input }) => {
24
- if (!ctx.user) {
25
- throw new TRPCError({
26
- code: "UNAUTHORIZED",
27
- message: "User must be authenticated",
28
- });
29
- }
30
- // Verify user is a teacher of the class
31
- const classData = await prisma.class.findFirst({
32
- where: {
33
- id: input.classId,
34
- teachers: {
35
- some: {
36
- id: ctx.user.id,
37
- },
38
- },
39
- },
40
- });
41
- if (!classData) {
42
- throw new TRPCError({
43
- code: "NOT_FOUND",
44
- message: "Class not found or you are not a teacher",
45
- });
46
- }
47
- const section = await prisma.section.create({
48
- data: {
49
- name: input.name,
50
- order: 0,
51
- class: {
52
- connect: { id: input.classId },
53
- },
54
- ...(input.color && {
55
- color: input.color,
56
- }),
57
- },
58
- });
59
- // Insert new section at top of unified list (sections + assignments) and normalize
60
- const [sections, assignments] = await Promise.all([
61
- prisma.section.findMany({
62
- where: { classId: input.classId },
63
- select: { id: true, order: true },
64
- }),
65
- prisma.assignment.findMany({
66
- where: { classId: input.classId },
67
- select: { id: true, order: true },
68
- }),
69
- ]);
70
- const unified = [
71
- ...sections.map(s => ({ id: s.id, order: s.order, type: 'section' })),
72
- ...assignments.map(a => ({ id: a.id, order: a.order, type: 'assignment' })),
73
- ].sort((a, b) => (a.order ?? Number.MAX_SAFE_INTEGER) - (b.order ?? Number.MAX_SAFE_INTEGER));
74
- const withoutNew = unified.filter(item => !(item.id === section.id && item.type === 'section'));
75
- const reindexed = [{ id: section.id, type: 'section' }, ...withoutNew.map(item => ({ id: item.id, type: item.type }))];
76
- await Promise.all(reindexed.map((item, index) => {
77
- if (item.type === 'section') {
78
- return prisma.section.update({ where: { id: item.id }, data: { order: index + 1 } });
79
- }
80
- else {
81
- return prisma.assignment.update({ where: { id: item.id }, data: { order: index + 1 } });
82
- }
83
- }));
84
- return section;
85
- }),
86
- reorder: protectedProcedure
28
+ .mutation(({ ctx, input }) => createSectionRecord(ctx.user.id, input)),
29
+ reorder: protectedTeacherProcedure
87
30
  .input(z.object({
88
31
  classId: z.string(),
89
- movedId: z.string(), // Section ID
90
- // One of: place at start/end of unified list, or relative to targetId (can be section or assignment)
91
- position: z.enum(['start', 'end', 'before', 'after']),
92
- targetId: z.string().optional(), // Can be a section ID or assignment ID
32
+ movedId: z.string(),
33
+ position: z.enum(["start", "end", "before", "after"]),
34
+ targetId: z.string().optional(),
93
35
  }))
94
- .mutation(async ({ ctx, input }) => {
95
- if (!ctx.user) {
96
- throw new TRPCError({
97
- code: "UNAUTHORIZED",
98
- message: "User must be authenticated",
99
- });
100
- }
101
- const { classId, movedId, position, targetId } = input;
102
- const moved = await prisma.section.findFirst({
103
- where: { id: movedId, classId },
104
- select: { id: true, classId: true },
105
- });
106
- if (!moved) {
107
- throw new TRPCError({ code: 'NOT_FOUND', message: 'Section not found' });
108
- }
109
- if ((position === 'before' || position === 'after') && !targetId) {
110
- throw new TRPCError({ code: 'BAD_REQUEST', message: 'targetId required for before/after' });
111
- }
112
- const result = await prisma.$transaction(async (tx) => {
113
- const [sections, assignments] = await Promise.all([
114
- tx.section.findMany({
115
- where: { classId },
116
- select: { id: true, order: true },
117
- }),
118
- tx.assignment.findMany({
119
- where: { classId },
120
- select: { id: true, order: true },
121
- }),
122
- ]);
123
- const unified = [
124
- ...sections.map(s => ({ id: s.id, order: s.order, type: 'section' })),
125
- ...assignments.map(a => ({ id: a.id, order: a.order, type: 'assignment' })),
126
- ].sort((a, b) => (a.order ?? Number.MAX_SAFE_INTEGER) - (b.order ?? Number.MAX_SAFE_INTEGER));
127
- const movedIdx = unified.findIndex(item => item.id === movedId && item.type === 'section');
128
- if (movedIdx === -1) {
129
- throw new TRPCError({ code: 'NOT_FOUND', message: 'Section not found in unified list' });
130
- }
131
- const withoutMoved = unified.filter(item => !(item.id === movedId && item.type === 'section'));
132
- let next = [];
133
- if (position === 'start') {
134
- next = [{ id: movedId, type: 'section' }, ...withoutMoved.map(item => ({ id: item.id, type: item.type }))];
135
- }
136
- else if (position === 'end') {
137
- next = [...withoutMoved.map(item => ({ id: item.id, type: item.type })), { id: movedId, type: 'section' }];
138
- }
139
- else {
140
- const targetIdx = withoutMoved.findIndex(item => item.id === targetId);
141
- if (targetIdx === -1) {
142
- throw new TRPCError({ code: 'BAD_REQUEST', message: 'targetId not found in unified list' });
143
- }
144
- if (position === 'before') {
145
- next = [
146
- ...withoutMoved.slice(0, targetIdx).map(item => ({ id: item.id, type: item.type })),
147
- { id: movedId, type: 'section' },
148
- ...withoutMoved.slice(targetIdx).map(item => ({ id: item.id, type: item.type })),
149
- ];
150
- }
151
- else {
152
- next = [
153
- ...withoutMoved.slice(0, targetIdx + 1).map(item => ({ id: item.id, type: item.type })),
154
- { id: movedId, type: 'section' },
155
- ...withoutMoved.slice(targetIdx + 1).map(item => ({ id: item.id, type: item.type })),
156
- ];
157
- }
158
- }
159
- // Normalize to 1..n
160
- await Promise.all(next.map((item, index) => {
161
- if (item.type === 'section') {
162
- return tx.section.update({ where: { id: item.id }, data: { order: index + 1 } });
163
- }
164
- else {
165
- return tx.assignment.update({ where: { id: item.id }, data: { order: index + 1 } });
166
- }
167
- }));
168
- return tx.section.findUnique({ where: { id: movedId } });
169
- });
170
- return result;
171
- }),
172
- update: protectedProcedure
36
+ .mutation(({ ctx, input }) => reorderSection(ctx.user.id, input)),
37
+ update: protectedTeacherProcedure
173
38
  .input(updateSectionSchema)
174
- .mutation(async ({ ctx, input }) => {
175
- if (!ctx.user) {
176
- throw new TRPCError({
177
- code: "UNAUTHORIZED",
178
- message: "User must be authenticated",
179
- });
180
- }
181
- // Verify user is a teacher of the class
182
- const classData = await prisma.class.findFirst({
183
- where: {
184
- id: input.classId,
185
- teachers: {
186
- some: {
187
- id: ctx.user.id,
188
- },
189
- },
190
- },
191
- });
192
- if (!classData) {
193
- throw new TRPCError({
194
- code: "NOT_FOUND",
195
- message: "Class not found or you are not a teacher",
196
- });
197
- }
198
- const section = await prisma.section.update({
199
- where: { id: input.id },
200
- data: {
201
- name: input.name,
202
- ...(input.color && {
203
- color: input.color,
204
- }),
205
- },
206
- });
207
- return section;
208
- }),
209
- reOrder: protectedProcedure
39
+ .mutation(({ ctx, input }) => updateSectionRecord(ctx.user.id, input)),
40
+ reOrder: protectedTeacherProcedure
210
41
  .input(z.object({
211
42
  id: z.string(),
212
43
  classId: z.string(),
213
44
  order: z.number(),
214
45
  }))
215
- .mutation(async ({ ctx, input }) => {
216
- if (!ctx.user) {
217
- throw new TRPCError({
218
- code: "UNAUTHORIZED",
219
- message: "User must be authenticated",
220
- });
221
- }
222
- // Verify user is a teacher of the class
223
- const classData = await prisma.class.findFirst({
224
- where: {
225
- id: input.classId,
226
- teachers: {
227
- some: {
228
- id: ctx.user.id,
229
- },
230
- },
231
- },
232
- });
233
- if (!classData) {
234
- throw new TRPCError({
235
- code: "NOT_FOUND",
236
- message: "Class not found or you are not a teacher",
237
- });
238
- }
239
- // Update order and normalize unified list
240
- await prisma.$transaction(async (tx) => {
241
- await tx.section.update({
242
- where: { id: input.id },
243
- data: { order: input.order },
244
- });
245
- // Normalize entire unified list
246
- const [sections, assignments] = await Promise.all([
247
- tx.section.findMany({
248
- where: { classId: input.classId },
249
- select: { id: true, order: true },
250
- }),
251
- tx.assignment.findMany({
252
- where: { classId: input.classId },
253
- select: { id: true, order: true },
254
- }),
255
- ]);
256
- const unified = [
257
- ...sections.map(s => ({ id: s.id, order: s.order, type: 'section' })),
258
- ...assignments.map(a => ({ id: a.id, order: a.order, type: 'assignment' })),
259
- ].sort((a, b) => (a.order ?? Number.MAX_SAFE_INTEGER) - (b.order ?? Number.MAX_SAFE_INTEGER));
260
- await Promise.all(unified.map((item, index) => {
261
- if (item.type === 'section') {
262
- return tx.section.update({ where: { id: item.id }, data: { order: index + 1 } });
263
- }
264
- else {
265
- return tx.assignment.update({ where: { id: item.id }, data: { order: index + 1 } });
266
- }
267
- }));
268
- });
269
- return { id: input.id };
270
- }),
271
- delete: protectedProcedure
46
+ .mutation(({ ctx, input }) => reOrderSection(ctx.user.id, input)),
47
+ delete: protectedTeacherProcedure
272
48
  .input(deleteSectionSchema)
273
- .mutation(async ({ ctx, input }) => {
274
- if (!ctx.user) {
275
- throw new TRPCError({
276
- code: "UNAUTHORIZED",
277
- message: "User must be authenticated",
278
- });
279
- }
280
- // Verify user is a teacher of the class
281
- const classData = await prisma.class.findFirst({
282
- where: {
283
- id: input.classId,
284
- teachers: {
285
- some: {
286
- id: ctx.user.id,
287
- },
288
- },
289
- },
290
- });
291
- if (!classData) {
292
- throw new TRPCError({
293
- code: "NOT_FOUND",
294
- message: "Class not found or you are not a teacher",
295
- });
296
- }
297
- await prisma.section.delete({
298
- where: { id: input.id },
299
- });
300
- return { id: input.id };
301
- }),
49
+ .mutation(({ ctx, input }) => deleteSectionRecord(ctx.user.id, input)),
302
50
  });
51
+ //# sourceMappingURL=section.js.map
52
+ //# debugId=d31ac70e-8c0c-5d80-9d4e-e996c6dc6b08
@@ -0,0 +1 @@
1
+ {"version":3,"file":"section.js","sources":["routers/section.ts"],"sourceRoot":"/","sourcesContent":["import { z } from \"zod\";\nimport {\n createTRPCRouter,\n protectedClassMemberProcedure,\n protectedTeacherProcedure,\n} from \"../trpc.js\";\nimport {\n sectionExists,\n createSectionRecord,\n reorderSection,\n updateSectionRecord,\n reOrderSection,\n deleteSectionRecord,\n} from \"../services/section.js\";\n\nconst createSectionSchema = z.object({\n classId: z.string(),\n id: z.string().optional(),\n name: z.string(),\n color: z.string().optional(),\n});\n\nconst updateSectionSchema = z.object({\n id: z.string(),\n classId: z.string(),\n name: z.string(),\n color: z.string().optional(),\n});\n\nconst deleteSectionSchema = z.object({\n id: z.string(),\n classId: z.string(),\n});\n\nexport const sectionRouter = createTRPCRouter({\n exists: protectedClassMemberProcedure\n .input(z.object({ id: z.string() }))\n .query(({ input }) => sectionExists(input.id)),\n\n create: protectedTeacherProcedure\n .input(createSectionSchema)\n .mutation(({ ctx, input }) =>\n createSectionRecord(ctx.user!.id, input)\n ),\n\n reorder: protectedTeacherProcedure\n .input(\n z.object({\n classId: z.string(),\n movedId: z.string(),\n position: z.enum([\"start\", \"end\", \"before\", \"after\"]),\n targetId: z.string().optional(),\n })\n )\n .mutation(({ ctx, input }) =>\n reorderSection(ctx.user!.id, input)\n ),\n\n update: protectedTeacherProcedure\n .input(updateSectionSchema)\n .mutation(({ ctx, input }) =>\n updateSectionRecord(ctx.user!.id, input)\n ),\n\n reOrder: protectedTeacherProcedure\n .input(\n z.object({\n id: z.string(),\n classId: z.string(),\n order: z.number(),\n })\n )\n .mutation(({ ctx, input }) =>\n reOrderSection(ctx.user!.id, input)\n ),\n\n delete: protectedTeacherProcedure\n .input(deleteSectionSchema)\n .mutation(({ ctx, input }) =>\n deleteSectionRecord(ctx.user!.id, input)\n ),\n});\n"],"names":[],"mappings":"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","debug_id":"d31ac70e-8c0c-5d80-9d4e-e996c6dc6b08"}
@@ -35,9 +35,9 @@ export declare const userRouter: import("@trpc/server").TRPCBuiltRouter<{
35
35
  updateProfile: import("@trpc/server").TRPCMutationProcedure<{
36
36
  input: {
37
37
  profile?: {
38
- location?: string | null | undefined;
39
38
  displayName?: string | null | undefined;
40
39
  bio?: string | null | undefined;
40
+ location?: string | null | undefined;
41
41
  website?: string | null | undefined;
42
42
  } | undefined;
43
43
  profilePicture?: {
@@ -1 +1 @@
1
- {"version":3,"file":"user.d.ts","sourceRoot":"","sources":["../../src/routers/user.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AA2DxB,eAAO,MAAM,UAAU;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAuMrB,CAAC"}
1
+ {"version":3,"file":"user.d.ts","sourceRoot":"/","sources":["routers/user.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AAiExB,eAAO,MAAM,UAAU;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAgBrB,CAAC"}