@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
@@ -0,0 +1,598 @@
1
+
2
+ /**
3
+ * Class model – classes, members, assignments, grades, mark schemes,
4
+ * grading boundaries, syllabus, lab drafts, invite codes.
5
+ */
6
+
7
+ import { Assignment, Class, Folder, GradingBoundary, MarkScheme, Section, Worksheet } from "@prisma/client";
8
+ import type { PrismaClient } from "@prisma/client";
9
+
10
+ import { prisma } from "../lib/prisma.js";
11
+ import { v4 as uuidv4 } from 'uuid';
12
+ import { copyFile } from "../lib/googleCloudStorage.js";
13
+
14
+ const classIncludeForGetAll = {
15
+ assignments: {
16
+ where: {
17
+ dueDate: { lte: new Date(new Date().setHours(23, 59, 59, 999)) },
18
+ template: false,
19
+ },
20
+ select: {
21
+ id: true,
22
+ title: true,
23
+ type: true,
24
+ dueDate: true,
25
+ },
26
+ },
27
+ students: {
28
+ select: {
29
+ id: true,
30
+ username: true,
31
+ profile: {
32
+ select: {
33
+ displayName: true,
34
+ profilePicture: true,
35
+ profilePictureThumbnail: true,
36
+ },
37
+ },
38
+ },
39
+ },
40
+ teachers: {
41
+ select: {
42
+ id: true,
43
+ username: true,
44
+ profile: {
45
+ select: {
46
+ displayName: true,
47
+ profilePicture: true,
48
+ profilePictureThumbnail: true,
49
+ },
50
+ },
51
+ },
52
+ },
53
+ } as const;
54
+
55
+ /** @returns Classes where user is a teacher. */
56
+ export function findTeacherClasses(userId: string) {
57
+ return prisma.class.findMany({
58
+ where: { teachers: { some: { id: userId } } },
59
+ include: classIncludeForGetAll,
60
+ });
61
+ }
62
+
63
+ /** @returns Classes where user is a student. */
64
+ export function findStudentClasses(userId: string) {
65
+ return prisma.class.findMany({
66
+ where: { students: { some: { id: userId } } },
67
+ include: classIncludeForGetAll,
68
+ });
69
+ }
70
+
71
+ /** @returns Class if user is a teacher. */
72
+ export function isTeacherInClass(classId: string, userId: string) {
73
+ return prisma.class.findFirst({
74
+ where: {
75
+ id: classId,
76
+ teachers: { some: { id: userId } },
77
+ },
78
+ });
79
+ }
80
+
81
+ const classIncludeForGet = {
82
+ teachers: {
83
+ select: {
84
+ id: true,
85
+ username: true,
86
+ profile: {
87
+ select: {
88
+ displayName: true,
89
+ profilePicture: true,
90
+ profilePictureThumbnail: true,
91
+ },
92
+ },
93
+ },
94
+ },
95
+ students: {
96
+ select: {
97
+ id: true,
98
+ username: true,
99
+ profile: {
100
+ select: {
101
+ displayName: true,
102
+ profilePicture: true,
103
+ profilePictureThumbnail: true,
104
+ },
105
+ },
106
+ },
107
+ },
108
+ announcements: {
109
+ orderBy: { createdAt: "desc" as const },
110
+ select: {
111
+ id: true,
112
+ remarks: true,
113
+ createdAt: true,
114
+ modifiedAt: true,
115
+ teacher: {
116
+ select: {
117
+ id: true,
118
+ username: true,
119
+ profile: {
120
+ select: {
121
+ displayName: true,
122
+ profilePicture: true,
123
+ profilePictureThumbnail: true,
124
+ },
125
+ },
126
+ },
127
+ },
128
+ },
129
+ },
130
+ assignments: {
131
+ select: {
132
+ type: true,
133
+ id: true,
134
+ title: true,
135
+ dueDate: true,
136
+ createdAt: true,
137
+ weight: true,
138
+ order: true,
139
+ graded: true,
140
+ maxGrade: true,
141
+ instructions: true,
142
+ inProgress: true,
143
+ section: {
144
+ select: { id: true, name: true },
145
+ },
146
+ markScheme: {
147
+ select: { id: true, structured: true },
148
+ },
149
+ gradingBoundary: {
150
+ select: { id: true, structured: true },
151
+ },
152
+ submissions: {
153
+ select: {
154
+ studentId: true,
155
+ id: true,
156
+ submitted: true,
157
+ gradeReceived: true,
158
+ rubricState: true,
159
+ teacherComments: true,
160
+ returned: true,
161
+ submittedAt: true,
162
+ },
163
+ },
164
+ },
165
+ },
166
+ } as const;
167
+
168
+ const assignmentSelect = {
169
+ type: true,
170
+ id: true,
171
+ title: true,
172
+ dueDate: true,
173
+ createdAt: true,
174
+ weight: true,
175
+ order: true,
176
+ graded: true,
177
+ maxGrade: true,
178
+ instructions: true,
179
+ inProgress: true,
180
+ section: { select: { id: true, name: true } },
181
+ markScheme: { select: { id: true, structured: true } },
182
+ gradingBoundary: { select: { id: true, structured: true } },
183
+ submissions: {
184
+ select: {
185
+ studentId: true,
186
+ id: true,
187
+ submitted: true,
188
+ gradeReceived: true,
189
+ rubricState: true,
190
+ teacherComments: true,
191
+ returned: true,
192
+ submittedAt: true,
193
+ },
194
+ },
195
+ } as const;
196
+
197
+ /** @returns Class with assignments. Students see only assigned; teachers see all. */
198
+ export function findClassWithAssignments(
199
+ classId: string,
200
+ options: { isTeacher: boolean; userId?: string }
201
+ ) {
202
+ const assignmentWhere = !options.isTeacher &&
203
+ options.userId
204
+ ? {
205
+ OR: [
206
+ { assignedTo: { some: { id: options.userId } } },
207
+ { assignedTo: { none: {} } },
208
+ ],
209
+ }
210
+ : undefined;
211
+
212
+ const submissionWhere =
213
+ !options.isTeacher && options.userId
214
+ ? { studentId: options.userId }
215
+ : undefined;
216
+
217
+ return prisma.class.findUnique({
218
+ where: { id: classId },
219
+ include: {
220
+ teachers: classIncludeForGet.teachers,
221
+ students: classIncludeForGet.students,
222
+ announcements: classIncludeForGet.announcements,
223
+ assignments: {
224
+ ...(assignmentWhere && { where: assignmentWhere }),
225
+ select: {
226
+ ...assignmentSelect,
227
+ submissions: {
228
+ ...(submissionWhere && { where: submissionWhere }),
229
+ select: assignmentSelect.submissions.select,
230
+ },
231
+ },
232
+ },
233
+ },
234
+ });
235
+ }
236
+
237
+ /** @returns Class by ID (minimal). */
238
+ export function findClassByIdBasic(classId: string) {
239
+ return prisma.class.findUnique({
240
+ where: { id: classId },
241
+ });
242
+ }
243
+
244
+ /** @returns Sections for a class. */
245
+ export function findSectionsByClassId(classId: string) {
246
+ return prisma.section.findMany({
247
+ where: { classId },
248
+ });
249
+ }
250
+
251
+ /** @returns User by ID (id, username only). */
252
+ export function findUserById(userId: string) {
253
+ return prisma.user.findUnique({
254
+ where: { id: userId },
255
+ select: { id: true, username: true },
256
+ });
257
+ }
258
+
259
+ /** @returns Session by class invite code. */
260
+ export function findSessionByClassCode(classCode: string) {
261
+ return prisma.session.findFirst({
262
+ where: {
263
+ id: { equals: classCode, mode: "insensitive" },
264
+ },
265
+ });
266
+ }
267
+
268
+ /** @returns Session for a class. */
269
+ export function findSessionByClassId(classId: string) {
270
+ return prisma.session.findFirst({
271
+ where: { classId },
272
+ });
273
+ }
274
+
275
+ /** @returns User's graded submissions for grade calculation. */
276
+ export function findSubmissionsForGrades(userId: string, classId: string) {
277
+ return prisma.submission.findMany({
278
+ where: {
279
+ studentId: userId,
280
+ assignment: { classId, graded: true },
281
+ },
282
+ include: {
283
+ assignment: {
284
+ select: {
285
+ id: true,
286
+ title: true,
287
+ maxGrade: true,
288
+ weight: true,
289
+ markSchemeId: true,
290
+ dueDate: true,
291
+ markScheme: { select: { structured: true } },
292
+ gradingBoundaryId: true,
293
+ gradingBoundary: { select: { structured: true } },
294
+ },
295
+ },
296
+ },
297
+ });
298
+ }
299
+
300
+ /** @returns Events for a class. */
301
+ export function findEventsByClassId(classId: string) {
302
+ return prisma.event.findMany({
303
+ where: { class: { id: classId } },
304
+ select: { name: true, startTime: true, endTime: true },
305
+ });
306
+ }
307
+
308
+ /** @returns Mark schemes for a class. */
309
+ export function findMarkSchemesByClassId(classId: string) {
310
+ return prisma.markScheme.findMany({
311
+ where: { class: { id: classId } },
312
+ });
313
+ }
314
+
315
+ /** @returns Grading boundaries for a class. */
316
+ export function findGradingBoundariesByClassId(classId: string) {
317
+ return prisma.gradingBoundary.findMany({
318
+ where: { class: { id: classId } },
319
+ });
320
+ }
321
+
322
+ /** @returns Class syllabus. */
323
+ export function findClassSyllabus(classId: string) {
324
+ return prisma.class.findUnique({
325
+ where: { id: classId },
326
+ select: { syllabus: true },
327
+ });
328
+ }
329
+
330
+ /** @returns In-progress lab drafts for a teacher. */
331
+ export function findLabDraftsByClass(classId: string, teacherId: string) {
332
+ return prisma.assignment.findMany({
333
+ where: {
334
+ classId,
335
+ teacherId,
336
+ inProgress: true,
337
+ },
338
+ orderBy: { modifiedAt: "desc" as const },
339
+ });
340
+ }
341
+
342
+ /** @returns Assignments with attachments and submissions. */
343
+ export function findAssignmentsWithFiles(classId: string) {
344
+ return prisma.assignment.findMany({
345
+ where: { classId },
346
+ include: {
347
+ attachments: {
348
+ select: {
349
+ id: true,
350
+ name: true,
351
+ type: true,
352
+ size: true,
353
+ path: true,
354
+ thumbnailId: true,
355
+ uploadedAt: true,
356
+ user: { select: { id: true, username: true } },
357
+ },
358
+ },
359
+ submissions: {
360
+ include: {
361
+ attachments: {
362
+ select: {
363
+ id: true,
364
+ name: true,
365
+ type: true,
366
+ size: true,
367
+ path: true,
368
+ thumbnailId: true,
369
+ uploadedAt: true,
370
+ user: { select: { id: true, username: true } },
371
+ },
372
+ },
373
+ annotations: {
374
+ select: {
375
+ id: true,
376
+ name: true,
377
+ type: true,
378
+ size: true,
379
+ path: true,
380
+ thumbnailId: true,
381
+ uploadedAt: true,
382
+ user: { select: { id: true, username: true } },
383
+ },
384
+ },
385
+ student: { select: { id: true, username: true } },
386
+ },
387
+ },
388
+ teacher: { select: { id: true, username: true } },
389
+ },
390
+ orderBy: { createdAt: "desc" as const },
391
+ });
392
+ }
393
+
394
+ async function recursivelyIncludeFiles(folderId: string) {
395
+ if (!folderId.length) {
396
+ return null;
397
+ }
398
+
399
+ const folder = await prisma.folder.findUnique({
400
+ where: { id: folderId },
401
+ include: {
402
+ files: {
403
+ select: { id: true, name: true, type: true, size: true, path: true },
404
+ },
405
+ childFolders: {
406
+ select: { id: true },
407
+ },
408
+ },
409
+ });
410
+ if (!folder) {
411
+ return null;
412
+ }
413
+ return {
414
+ ...folder,
415
+ files: folder.files.map(file => file.id),
416
+ childFolders: await Promise.all(folder.childFolders.map(folder => recursivelyIncludeFiles(folder.id))),
417
+ };
418
+ }
419
+
420
+ async function recursivelyCreateFiles(
421
+ tx: PrismaClient | Omit<PrismaClient, "$connect" | "$disconnect" | "$on" | "$transaction" | "$use" | "$extends">,
422
+ folderId: string,
423
+ targetFolderId?: string
424
+ ) {
425
+ if (!folderId.length) {
426
+ return null;
427
+ }
428
+
429
+ const parentFolder = await tx.folder.findUnique({
430
+ where: { id: folderId },
431
+ include: {
432
+ files: {
433
+ select: { id: true, name: true, type: true, size: true, path: true },
434
+ },
435
+ childFolders: {
436
+ select: { id: true, name: true, color: true, parentFolderId: true },
437
+ },
438
+ },
439
+ });
440
+ if (!parentFolder) {
441
+ return null;
442
+ }
443
+
444
+ const createdFiles: { count: number } = { count: 0 };
445
+ for (const file of parentFolder.files) {
446
+ const ext = file.name.split('.').pop() || '';
447
+ const newPath = `imported/${uuidv4()}${ext ? `.${ext}` : ''}`;
448
+ await copyFile(file.path, newPath);
449
+ await tx.file.create({
450
+ data: {
451
+ name: file.name,
452
+ type: file.type,
453
+ size: file.size,
454
+ path: newPath,
455
+ uploadStatus: 'COMPLETED',
456
+ ...(targetFolderId && { folder: { connect: { id: targetFolderId } } }),
457
+ },
458
+ });
459
+ createdFiles.count += 1;
460
+ }
461
+
462
+ const childFolders: { files: { count: number }; childFolders: unknown[] }[] = [];
463
+ for (const childFolder of parentFolder.childFolders) {
464
+ const newFolder = await tx.folder.create({
465
+ data: {
466
+ name: childFolder.name,
467
+ color: childFolder.color,
468
+ parentFolderId: targetFolderId ?? parentFolder.id,
469
+ },
470
+ });
471
+ const result = await recursivelyCreateFiles(tx, childFolder.id, newFolder.id);
472
+ if (result) childFolders.push(result);
473
+ }
474
+
475
+ return {
476
+ files: createdFiles,
477
+ childFolders,
478
+ };
479
+ }
480
+
481
+
482
+ export async function findFullExportableClass(classId: string) {
483
+ const baseClass = await prisma.class.findUnique({
484
+ where: { id: classId },
485
+ include: {
486
+ assignments: {
487
+ select: {
488
+ ...assignmentSelect,
489
+ submissions: false,
490
+ },
491
+ },
492
+ classFiles: {
493
+ select: {
494
+ id: true,
495
+ },
496
+ },
497
+ worksheets: true,
498
+ markSchemes: true,
499
+ gradingBoundaries: true,
500
+ sections: true,
501
+ },
502
+ });
503
+
504
+ if (!baseClass) {
505
+ return null;
506
+ }
507
+
508
+ const classFiles = await recursivelyIncludeFiles(baseClass.classFiles?.id ?? "");
509
+
510
+ const formattedClass = {
511
+ ...baseClass,
512
+ classFiles,
513
+ };
514
+
515
+ return formattedClass;
516
+ }
517
+
518
+ export async function createClassByImport(classId: string, userId: string, year: number, classData: Class & { classFiles: Folder | null }): Promise<string | null> {
519
+ const newClassId = await prisma.$transaction(async (tx) => {
520
+ const createdClass = await tx.class.create({
521
+ data: {
522
+ name: classData.name,
523
+ subject: classData.subject,
524
+ color: classData.color,
525
+ section: classData.section,
526
+ syllabus: classData.syllabus,
527
+ teachers: { connect: { id: userId } },
528
+ classFiles: { create: { name: "Class Files" } },
529
+ },
530
+ include: {
531
+ classFiles: true,
532
+ },
533
+ });
534
+
535
+ const assignments = await prisma.assignment.createMany({
536
+ data: (classData as unknown as Class & { assignments: Assignment[] }).assignments.map((assignment) => {
537
+ const newDate = new Date(assignment.dueDate).setFullYear(year);
538
+ return {
539
+ ...assignment,
540
+ id: assignment.id,
541
+ title: assignment.title,
542
+ type: assignment.type,
543
+ dueDate: newDate as unknown as Date,
544
+ };
545
+ })
546
+ });
547
+
548
+ const worksheets = await prisma.worksheet.createMany({
549
+ data: (classData as unknown as Class & { worksheets: Worksheet[] }).worksheets.map((worksheet) => {
550
+ return {
551
+ ...worksheet,
552
+ id: worksheet.id,
553
+ name: worksheet.name,
554
+ };
555
+ })
556
+ });
557
+
558
+ const markSchemes = await prisma.markScheme.createMany({
559
+ data: (classData as unknown as Class & { markSchemes: MarkScheme[] }).markSchemes.map((markScheme) => {
560
+ return {
561
+ ...markScheme,
562
+ id: markScheme.id,
563
+ structured: markScheme.structured,
564
+ };
565
+ })
566
+ });
567
+
568
+ const gradingBoundaries = await prisma.gradingBoundary.createMany({
569
+ data: (classData as unknown as Class & { gradingBoundaries: GradingBoundary[] }).gradingBoundaries.map((gradingBoundary) => {
570
+ return {
571
+ ...gradingBoundary,
572
+ id: gradingBoundary.id,
573
+ structured: gradingBoundary.structured,
574
+ };
575
+ })
576
+ });
577
+
578
+ const sections = await prisma.section.createMany({
579
+ data: (classData as unknown as Class & { sections: Section[] }).sections.map((section) => {
580
+ return {
581
+ ...section,
582
+ id: section.id,
583
+ name: section.name,
584
+ };
585
+ })
586
+ });
587
+
588
+ const classFiles = await recursivelyCreateFiles(tx, classData.classFiles?.id ?? uuidv4(), createdClass.classFiles?.id);
589
+ if (!classFiles) {
590
+ return null;
591
+ }
592
+
593
+ return createdClass.id;
594
+
595
+ });
596
+
597
+ return newClassId;
598
+ }