@hed-hog/lms 0.0.349 → 0.0.350

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 (409) hide show
  1. package/dist/achievement/achievement.controller.d.ts +62 -0
  2. package/dist/achievement/achievement.controller.d.ts.map +1 -0
  3. package/dist/achievement/achievement.controller.js +90 -0
  4. package/dist/achievement/achievement.controller.js.map +1 -0
  5. package/dist/achievement/achievement.mcp-tools.d.ts +19 -0
  6. package/dist/achievement/achievement.mcp-tools.d.ts.map +1 -0
  7. package/dist/achievement/achievement.mcp-tools.js +157 -0
  8. package/dist/achievement/achievement.mcp-tools.js.map +1 -0
  9. package/dist/achievement/achievement.module.d.ts +3 -0
  10. package/dist/achievement/achievement.module.d.ts.map +1 -0
  11. package/dist/achievement/achievement.module.js +26 -0
  12. package/dist/achievement/achievement.module.js.map +1 -0
  13. package/dist/achievement/achievement.service.d.ts +72 -0
  14. package/dist/achievement/achievement.service.d.ts.map +1 -0
  15. package/dist/achievement/achievement.service.js +200 -0
  16. package/dist/achievement/achievement.service.js.map +1 -0
  17. package/dist/achievement/dto/create-achievement.dto.d.ts +12 -0
  18. package/dist/achievement/dto/create-achievement.dto.d.ts.map +1 -0
  19. package/dist/achievement/dto/create-achievement.dto.js +60 -0
  20. package/dist/achievement/dto/create-achievement.dto.js.map +1 -0
  21. package/dist/achievement/dto/update-achievement.dto.d.ts +11 -0
  22. package/dist/achievement/dto/update-achievement.dto.d.ts.map +1 -0
  23. package/dist/achievement/dto/update-achievement.dto.js +57 -0
  24. package/dist/achievement/dto/update-achievement.dto.js.map +1 -0
  25. package/dist/bitcode-wallet/bitcode-wallet.controller.d.ts +114 -0
  26. package/dist/bitcode-wallet/bitcode-wallet.controller.d.ts.map +1 -0
  27. package/dist/bitcode-wallet/bitcode-wallet.controller.js +102 -0
  28. package/dist/bitcode-wallet/bitcode-wallet.controller.js.map +1 -0
  29. package/dist/bitcode-wallet/bitcode-wallet.mcp-tools.d.ts +25 -0
  30. package/dist/bitcode-wallet/bitcode-wallet.mcp-tools.d.ts.map +1 -0
  31. package/dist/bitcode-wallet/bitcode-wallet.mcp-tools.js +160 -0
  32. package/dist/bitcode-wallet/bitcode-wallet.mcp-tools.js.map +1 -0
  33. package/dist/bitcode-wallet/bitcode-wallet.module.d.ts +3 -0
  34. package/dist/bitcode-wallet/bitcode-wallet.module.d.ts.map +1 -0
  35. package/dist/bitcode-wallet/bitcode-wallet.module.js +26 -0
  36. package/dist/bitcode-wallet/bitcode-wallet.module.js.map +1 -0
  37. package/dist/bitcode-wallet/bitcode-wallet.service.d.ts +127 -0
  38. package/dist/bitcode-wallet/bitcode-wallet.service.d.ts.map +1 -0
  39. package/dist/bitcode-wallet/bitcode-wallet.service.js +264 -0
  40. package/dist/bitcode-wallet/bitcode-wallet.service.js.map +1 -0
  41. package/dist/bitcode-wallet/dto/create-bitcode-wallet-transaction.dto.d.ts +8 -0
  42. package/dist/bitcode-wallet/dto/create-bitcode-wallet-transaction.dto.d.ts.map +1 -0
  43. package/dist/bitcode-wallet/dto/create-bitcode-wallet-transaction.dto.js +33 -0
  44. package/dist/bitcode-wallet/dto/create-bitcode-wallet-transaction.dto.js.map +1 -0
  45. package/dist/bitcode-wallet/dto/create-bitcode-wallet.dto.d.ts +4 -0
  46. package/dist/bitcode-wallet/dto/create-bitcode-wallet.dto.d.ts.map +1 -0
  47. package/dist/bitcode-wallet/dto/create-bitcode-wallet.dto.js +22 -0
  48. package/dist/bitcode-wallet/dto/create-bitcode-wallet.dto.js.map +1 -0
  49. package/dist/bitcode-wallet/dto/update-bitcode-wallet-transaction.dto.d.ts +7 -0
  50. package/dist/bitcode-wallet/dto/update-bitcode-wallet-transaction.dto.d.ts.map +1 -0
  51. package/dist/bitcode-wallet/dto/update-bitcode-wallet-transaction.dto.js +35 -0
  52. package/dist/bitcode-wallet/dto/update-bitcode-wallet-transaction.dto.js.map +1 -0
  53. package/dist/bitcode-wallet/dto/update-bitcode-wallet.dto.d.ts +4 -0
  54. package/dist/bitcode-wallet/dto/update-bitcode-wallet.dto.d.ts.map +1 -0
  55. package/dist/bitcode-wallet/dto/update-bitcode-wallet.dto.js +23 -0
  56. package/dist/bitcode-wallet/dto/update-bitcode-wallet.dto.js.map +1 -0
  57. package/dist/certificate/certificate.controller.d.ts +22 -0
  58. package/dist/certificate/certificate.controller.d.ts.map +1 -1
  59. package/dist/certificate/certificate.controller.js +12 -0
  60. package/dist/certificate/certificate.controller.js.map +1 -1
  61. package/dist/certificate/certificate.mcp-tools.d.ts +24 -0
  62. package/dist/certificate/certificate.mcp-tools.d.ts.map +1 -0
  63. package/dist/certificate/certificate.mcp-tools.js +188 -0
  64. package/dist/certificate/certificate.mcp-tools.js.map +1 -0
  65. package/dist/certificate/certificate.module.d.ts.map +1 -1
  66. package/dist/certificate/certificate.module.js +2 -1
  67. package/dist/certificate/certificate.module.js.map +1 -1
  68. package/dist/certificate/certificate.service.d.ts +25 -2
  69. package/dist/certificate/certificate.service.d.ts.map +1 -1
  70. package/dist/certificate/certificate.service.js +87 -2
  71. package/dist/certificate/certificate.service.js.map +1 -1
  72. package/dist/certificate/dto/update-certificate-public-access.dto.d.ts +4 -0
  73. package/dist/certificate/dto/update-certificate-public-access.dto.d.ts.map +1 -0
  74. package/dist/certificate/dto/update-certificate-public-access.dto.js +21 -0
  75. package/dist/certificate/dto/update-certificate-public-access.dto.js.map +1 -0
  76. package/dist/class-group/class-group.mcp-tools.d.ts +87 -0
  77. package/dist/class-group/class-group.mcp-tools.d.ts.map +1 -0
  78. package/dist/class-group/class-group.mcp-tools.js +553 -0
  79. package/dist/class-group/class-group.mcp-tools.js.map +1 -0
  80. package/dist/class-group/class-group.module.d.ts.map +1 -1
  81. package/dist/class-group/class-group.module.js +2 -1
  82. package/dist/class-group/class-group.module.js.map +1 -1
  83. package/dist/class-group/class-group.service.d.ts +3 -1
  84. package/dist/class-group/class-group.service.d.ts.map +1 -1
  85. package/dist/class-group/class-group.service.js +45 -2
  86. package/dist/class-group/class-group.service.js.map +1 -1
  87. package/dist/course/course-operations-integration.service.d.ts +40 -0
  88. package/dist/course/course-operations-integration.service.d.ts.map +1 -0
  89. package/dist/course/course-operations-integration.service.js +372 -0
  90. package/dist/course/course-operations-integration.service.js.map +1 -0
  91. package/dist/course/course-structure.controller.d.ts +43 -4
  92. package/dist/course/course-structure.controller.d.ts.map +1 -1
  93. package/dist/course/course-structure.controller.js +22 -0
  94. package/dist/course/course-structure.controller.js.map +1 -1
  95. package/dist/course/course-structure.service.d.ts +42 -1
  96. package/dist/course/course-structure.service.d.ts.map +1 -1
  97. package/dist/course/course-structure.service.js +199 -32
  98. package/dist/course/course-structure.service.js.map +1 -1
  99. package/dist/course/course.controller.d.ts +12 -0
  100. package/dist/course/course.controller.d.ts.map +1 -1
  101. package/dist/course/course.mcp-tools.d.ts +90 -0
  102. package/dist/course/course.mcp-tools.d.ts.map +1 -0
  103. package/dist/course/course.mcp-tools.js +520 -0
  104. package/dist/course/course.mcp-tools.js.map +1 -0
  105. package/dist/course/course.module.d.ts.map +1 -1
  106. package/dist/course/course.module.js +8 -1
  107. package/dist/course/course.module.js.map +1 -1
  108. package/dist/course/course.service.d.ts +15 -1
  109. package/dist/course/course.service.d.ts.map +1 -1
  110. package/dist/course/course.service.js +70 -35
  111. package/dist/course/course.service.js.map +1 -1
  112. package/dist/course/dto/create-course.dto.d.ts +1 -0
  113. package/dist/course/dto/create-course.dto.d.ts.map +1 -1
  114. package/dist/course/dto/create-course.dto.js +7 -0
  115. package/dist/course/dto/create-course.dto.js.map +1 -1
  116. package/dist/course/dto/update-course-resources.dto.d.ts +11 -0
  117. package/dist/course/dto/update-course-resources.dto.d.ts.map +1 -0
  118. package/dist/course/dto/update-course-resources.dto.js +51 -0
  119. package/dist/course/dto/update-course-resources.dto.js.map +1 -0
  120. package/dist/course-lesson-discussion/course-lesson-discussion.controller.d.ts +23 -0
  121. package/dist/course-lesson-discussion/course-lesson-discussion.controller.d.ts.map +1 -0
  122. package/dist/course-lesson-discussion/course-lesson-discussion.controller.js +78 -0
  123. package/dist/course-lesson-discussion/course-lesson-discussion.controller.js.map +1 -0
  124. package/dist/course-lesson-discussion/course-lesson-discussion.mcp-tools.d.ts +22 -0
  125. package/dist/course-lesson-discussion/course-lesson-discussion.mcp-tools.d.ts.map +1 -0
  126. package/dist/course-lesson-discussion/course-lesson-discussion.mcp-tools.js +120 -0
  127. package/dist/course-lesson-discussion/course-lesson-discussion.mcp-tools.js.map +1 -0
  128. package/dist/course-lesson-discussion/course-lesson-discussion.module.d.ts +3 -0
  129. package/dist/course-lesson-discussion/course-lesson-discussion.module.d.ts.map +1 -0
  130. package/dist/course-lesson-discussion/course-lesson-discussion.module.js +26 -0
  131. package/dist/course-lesson-discussion/course-lesson-discussion.module.js.map +1 -0
  132. package/dist/course-lesson-discussion/course-lesson-discussion.service.d.ts +49 -0
  133. package/dist/course-lesson-discussion/course-lesson-discussion.service.d.ts.map +1 -0
  134. package/dist/course-lesson-discussion/course-lesson-discussion.service.js +272 -0
  135. package/dist/course-lesson-discussion/course-lesson-discussion.service.js.map +1 -0
  136. package/dist/course-lesson-discussion/dto/create-course-lesson-discussion-topic.dto.d.ts +6 -0
  137. package/dist/course-lesson-discussion/dto/create-course-lesson-discussion-topic.dto.d.ts.map +1 -0
  138. package/dist/course-lesson-discussion/dto/create-course-lesson-discussion-topic.dto.js +33 -0
  139. package/dist/course-lesson-discussion/dto/create-course-lesson-discussion-topic.dto.js.map +1 -0
  140. package/dist/course-lesson-note/course-lesson-note.controller.d.ts +53 -0
  141. package/dist/course-lesson-note/course-lesson-note.controller.d.ts.map +1 -0
  142. package/dist/course-lesson-note/course-lesson-note.controller.js +93 -0
  143. package/dist/course-lesson-note/course-lesson-note.controller.js.map +1 -0
  144. package/dist/course-lesson-note/course-lesson-note.mcp-tools.d.ts +27 -0
  145. package/dist/course-lesson-note/course-lesson-note.mcp-tools.d.ts.map +1 -0
  146. package/dist/course-lesson-note/course-lesson-note.mcp-tools.js +145 -0
  147. package/dist/course-lesson-note/course-lesson-note.mcp-tools.js.map +1 -0
  148. package/dist/course-lesson-note/course-lesson-note.module.d.ts +3 -0
  149. package/dist/course-lesson-note/course-lesson-note.module.d.ts.map +1 -0
  150. package/dist/course-lesson-note/course-lesson-note.module.js +26 -0
  151. package/dist/course-lesson-note/course-lesson-note.module.js.map +1 -0
  152. package/dist/course-lesson-note/course-lesson-note.service.d.ts +59 -0
  153. package/dist/course-lesson-note/course-lesson-note.service.d.ts.map +1 -0
  154. package/dist/course-lesson-note/course-lesson-note.service.js +195 -0
  155. package/dist/course-lesson-note/course-lesson-note.service.js.map +1 -0
  156. package/dist/course-lesson-note/dto/create-course-lesson-note.dto.d.ts +6 -0
  157. package/dist/course-lesson-note/dto/create-course-lesson-note.dto.d.ts.map +1 -0
  158. package/dist/course-lesson-note/dto/create-course-lesson-note.dto.js +33 -0
  159. package/dist/course-lesson-note/dto/create-course-lesson-note.dto.js.map +1 -0
  160. package/dist/course-lesson-note/dto/update-course-lesson-note.dto.d.ts +6 -0
  161. package/dist/course-lesson-note/dto/update-course-lesson-note.dto.d.ts.map +1 -0
  162. package/dist/course-lesson-note/dto/update-course-lesson-note.dto.js +35 -0
  163. package/dist/course-lesson-note/dto/update-course-lesson-note.dto.js.map +1 -0
  164. package/dist/dashboard/dashboard.mcp-tools.d.ts +10 -0
  165. package/dist/dashboard/dashboard.mcp-tools.d.ts.map +1 -0
  166. package/dist/dashboard/dashboard.mcp-tools.js +46 -0
  167. package/dist/dashboard/dashboard.mcp-tools.js.map +1 -0
  168. package/dist/dashboard/dashboard.module.d.ts.map +1 -1
  169. package/dist/dashboard/dashboard.module.js +2 -1
  170. package/dist/dashboard/dashboard.module.js.map +1 -1
  171. package/dist/enterprise/enterprise.mcp-tools.d.ts +82 -0
  172. package/dist/enterprise/enterprise.mcp-tools.d.ts.map +1 -0
  173. package/dist/enterprise/enterprise.mcp-tools.js +516 -0
  174. package/dist/enterprise/enterprise.mcp-tools.js.map +1 -0
  175. package/dist/enterprise/enterprise.module.d.ts.map +1 -1
  176. package/dist/enterprise/enterprise.module.js +2 -1
  177. package/dist/enterprise/enterprise.module.js.map +1 -1
  178. package/dist/enterprise/training/enterprise-training.module.d.ts.map +1 -1
  179. package/dist/enterprise/training/enterprise-training.module.js +11 -1
  180. package/dist/enterprise/training/enterprise-training.module.js.map +1 -1
  181. package/dist/enterprise/training/training-admin.mcp-tools.d.ts +79 -0
  182. package/dist/enterprise/training/training-admin.mcp-tools.d.ts.map +1 -0
  183. package/dist/enterprise/training/training-admin.mcp-tools.js +620 -0
  184. package/dist/enterprise/training/training-admin.mcp-tools.js.map +1 -0
  185. package/dist/enterprise/training/training-instructor.mcp-tools.d.ts +47 -0
  186. package/dist/enterprise/training/training-instructor.mcp-tools.d.ts.map +1 -0
  187. package/dist/enterprise/training/training-instructor.mcp-tools.js +275 -0
  188. package/dist/enterprise/training/training-instructor.mcp-tools.js.map +1 -0
  189. package/dist/enterprise/training/training-student.controller.d.ts +24 -0
  190. package/dist/enterprise/training/training-student.controller.d.ts.map +1 -1
  191. package/dist/enterprise/training/training-student.controller.js +22 -0
  192. package/dist/enterprise/training/training-student.controller.js.map +1 -1
  193. package/dist/enterprise/training/training-student.mcp-tools.d.ts +27 -0
  194. package/dist/enterprise/training/training-student.mcp-tools.d.ts.map +1 -0
  195. package/dist/enterprise/training/training-student.mcp-tools.js +186 -0
  196. package/dist/enterprise/training/training-student.mcp-tools.js.map +1 -0
  197. package/dist/enterprise/training/training-student.service.d.ts +32 -0
  198. package/dist/enterprise/training/training-student.service.d.ts.map +1 -1
  199. package/dist/enterprise/training/training-student.service.js +138 -0
  200. package/dist/enterprise/training/training-student.service.js.map +1 -1
  201. package/dist/evaluation/evaluation.mcp-tools.d.ts +25 -0
  202. package/dist/evaluation/evaluation.mcp-tools.d.ts.map +1 -0
  203. package/dist/evaluation/evaluation.mcp-tools.js +220 -0
  204. package/dist/evaluation/evaluation.mcp-tools.js.map +1 -0
  205. package/dist/evaluation/evaluation.module.d.ts.map +1 -1
  206. package/dist/evaluation/evaluation.module.js +2 -1
  207. package/dist/evaluation/evaluation.module.js.map +1 -1
  208. package/dist/exam/dto/create-exam-question.dto.d.ts +2 -0
  209. package/dist/exam/dto/create-exam-question.dto.d.ts.map +1 -1
  210. package/dist/exam/dto/create-exam-question.dto.js +10 -0
  211. package/dist/exam/dto/create-exam-question.dto.js.map +1 -1
  212. package/dist/exam/dto/create-exam.dto.d.ts +2 -0
  213. package/dist/exam/dto/create-exam.dto.d.ts.map +1 -1
  214. package/dist/exam/dto/create-exam.dto.js +10 -0
  215. package/dist/exam/dto/create-exam.dto.js.map +1 -1
  216. package/dist/exam/dto/create-question-subject.dto.d.ts +5 -0
  217. package/dist/exam/dto/create-question-subject.dto.d.ts.map +1 -0
  218. package/dist/exam/dto/create-question-subject.dto.js +28 -0
  219. package/dist/exam/dto/create-question-subject.dto.js.map +1 -0
  220. package/dist/exam/exam-attempt.controller.d.ts +4 -0
  221. package/dist/exam/exam-attempt.controller.d.ts.map +1 -1
  222. package/dist/exam/exam-attempt.service.d.ts +7 -1
  223. package/dist/exam/exam-attempt.service.d.ts.map +1 -1
  224. package/dist/exam/exam-attempt.service.js +47 -17
  225. package/dist/exam/exam-attempt.service.js.map +1 -1
  226. package/dist/exam/exam.controller.d.ts +34 -0
  227. package/dist/exam/exam.controller.d.ts.map +1 -1
  228. package/dist/exam/exam.controller.js +27 -0
  229. package/dist/exam/exam.controller.js.map +1 -1
  230. package/dist/exam/exam.mcp-tools.d.ts +62 -0
  231. package/dist/exam/exam.mcp-tools.d.ts.map +1 -0
  232. package/dist/exam/exam.mcp-tools.js +430 -0
  233. package/dist/exam/exam.mcp-tools.js.map +1 -0
  234. package/dist/exam/exam.module.d.ts.map +1 -1
  235. package/dist/exam/exam.module.js +2 -1
  236. package/dist/exam/exam.module.js.map +1 -1
  237. package/dist/exam/exam.service.d.ts +38 -0
  238. package/dist/exam/exam.service.d.ts.map +1 -1
  239. package/dist/exam/exam.service.js +114 -17
  240. package/dist/exam/exam.service.js.map +1 -1
  241. package/dist/index.d.ts +9 -0
  242. package/dist/index.d.ts.map +1 -1
  243. package/dist/index.js +9 -0
  244. package/dist/index.js.map +1 -1
  245. package/dist/instructor/instructor.mcp-tools.d.ts +41 -0
  246. package/dist/instructor/instructor.mcp-tools.d.ts.map +1 -0
  247. package/dist/instructor/instructor.mcp-tools.js +326 -0
  248. package/dist/instructor/instructor.mcp-tools.js.map +1 -0
  249. package/dist/instructor/instructor.module.d.ts.map +1 -1
  250. package/dist/instructor/instructor.module.js +2 -1
  251. package/dist/instructor/instructor.module.js.map +1 -1
  252. package/dist/lms.module.d.ts.map +1 -1
  253. package/dist/lms.module.js +15 -0
  254. package/dist/lms.module.js.map +1 -1
  255. package/dist/realtime/lms-realtime.controller.d.ts +7 -0
  256. package/dist/realtime/lms-realtime.controller.d.ts.map +1 -0
  257. package/dist/realtime/lms-realtime.controller.js +34 -0
  258. package/dist/realtime/lms-realtime.controller.js.map +1 -0
  259. package/dist/realtime/lms-realtime.module.d.ts +3 -0
  260. package/dist/realtime/lms-realtime.module.d.ts.map +1 -0
  261. package/dist/realtime/lms-realtime.module.js +25 -0
  262. package/dist/realtime/lms-realtime.module.js.map +1 -0
  263. package/dist/realtime/lms-realtime.service.d.ts +36 -0
  264. package/dist/realtime/lms-realtime.service.d.ts.map +1 -0
  265. package/dist/realtime/lms-realtime.service.js +59 -0
  266. package/dist/realtime/lms-realtime.service.js.map +1 -0
  267. package/dist/realtime/lms-realtime.subscriber.d.ts +10 -0
  268. package/dist/realtime/lms-realtime.subscriber.d.ts.map +1 -0
  269. package/dist/realtime/lms-realtime.subscriber.js +70 -0
  270. package/dist/realtime/lms-realtime.subscriber.js.map +1 -0
  271. package/dist/reports/reports.mcp-tools.d.ts +10 -0
  272. package/dist/reports/reports.mcp-tools.d.ts.map +1 -0
  273. package/dist/reports/reports.mcp-tools.js +50 -0
  274. package/dist/reports/reports.mcp-tools.js.map +1 -0
  275. package/dist/reports/reports.module.d.ts.map +1 -1
  276. package/dist/reports/reports.module.js +2 -1
  277. package/dist/reports/reports.module.js.map +1 -1
  278. package/dist/training/training.mcp-tools.d.ts +20 -0
  279. package/dist/training/training.mcp-tools.d.ts.map +1 -0
  280. package/dist/training/training.mcp-tools.js +181 -0
  281. package/dist/training/training.mcp-tools.js.map +1 -0
  282. package/dist/training/training.module.d.ts.map +1 -1
  283. package/dist/training/training.module.js +2 -1
  284. package/dist/training/training.module.js.map +1 -1
  285. package/hedhog/data/integration_event_catalog.yaml +69 -0
  286. package/hedhog/data/menu.yaml +34 -0
  287. package/hedhog/data/route.yaml +2351 -0
  288. package/hedhog/frontend/app/_components/class-form-sheet.tsx.ejs +168 -103
  289. package/hedhog/frontend/app/_components/course-form-sheet.tsx.ejs +80 -1
  290. package/hedhog/frontend/app/_components/course-picker.tsx.ejs +228 -0
  291. package/hedhog/frontend/app/_lib/hooks/use-lms-realtime-refresh.ts.ejs +58 -0
  292. package/hedhog/frontend/app/achievements/page.tsx.ejs +844 -0
  293. package/hedhog/frontend/app/bitcodes/page.tsx.ejs +1010 -0
  294. package/hedhog/frontend/app/certificates/issued/page.tsx.ejs +61 -2
  295. package/hedhog/frontend/app/classes/[id]/page.tsx.ejs +7 -0
  296. package/hedhog/frontend/app/classes/page.tsx.ejs +55 -2060
  297. package/hedhog/frontend/app/courses/[id]/_components/CourseRelationsCard.tsx.ejs +28 -0
  298. package/hedhog/frontend/app/courses/[id]/_components/course-edit-types.ts.ejs +1 -0
  299. package/hedhog/frontend/app/courses/[id]/page.tsx.ejs +0 -9
  300. package/hedhog/frontend/app/courses/[id]/structure/_components/course-tree-dnd.tsx.ejs +80 -66
  301. package/hedhog/frontend/app/courses/[id]/structure/_components/editor-course.tsx.ejs +583 -8
  302. package/hedhog/frontend/app/courses/[id]/structure/_components/editor-lesson.tsx.ejs +527 -57
  303. package/hedhog/frontend/app/courses/[id]/structure/_components/mock-data.ts.ejs +1 -1
  304. package/hedhog/frontend/app/courses/[id]/structure/_components/tree-context-menu.tsx.ejs +106 -4
  305. package/hedhog/frontend/app/courses/[id]/structure/_components/types.ts.ejs +2 -1
  306. package/hedhog/frontend/app/courses/[id]/structure/_data/adapters/course-structure.adapter.ts.ejs +11 -1
  307. package/hedhog/frontend/app/courses/[id]/structure/_data/services/course-structure.service.ts.ejs +53 -6
  308. package/hedhog/frontend/app/courses/[id]/structure/_data/types/api-course.types.ts.ejs +13 -2
  309. package/hedhog/frontend/app/courses/page.tsx.ejs +175 -29
  310. package/hedhog/frontend/app/enterprise/_components/enterprise-course-create-sheet.tsx.ejs +3 -0
  311. package/hedhog/frontend/app/enterprise/_components/enterprise-course-edit-sheet.tsx.ejs +7 -0
  312. package/hedhog/frontend/app/exams/[id]/questions/page.tsx.ejs +169 -2
  313. package/hedhog/frontend/app/exams/page.tsx.ejs +77 -22
  314. package/hedhog/frontend/app/instructor-skills/page.tsx.ejs +1 -0
  315. package/hedhog/frontend/app/instructors/page.tsx.ejs +1 -0
  316. package/hedhog/frontend/app/paths/page.tsx.ejs +6 -24
  317. package/hedhog/frontend/app/training/page.tsx.ejs +6 -24
  318. package/hedhog/frontend/messages/en.json +314 -12
  319. package/hedhog/frontend/messages/pt.json +314 -12
  320. package/hedhog/query/triggers.sql +53 -0
  321. package/hedhog/table/achievement.yaml +46 -0
  322. package/hedhog/table/bitcode_wallet.yaml +18 -0
  323. package/hedhog/table/bitcode_wallet_transaction.yaml +22 -0
  324. package/hedhog/table/certificate.yaml +3 -0
  325. package/hedhog/table/course.yaml +4 -0
  326. package/hedhog/table/course_file.yaml +23 -0
  327. package/hedhog/table/course_lesson.yaml +5 -0
  328. package/hedhog/table/course_lesson_discussion_like.yaml +21 -0
  329. package/hedhog/table/course_lesson_discussion_topic.yaml +35 -0
  330. package/hedhog/table/course_lesson_note.yaml +34 -0
  331. package/hedhog/table/exam.yaml +5 -0
  332. package/hedhog/table/learning_path_enrollment.yaml +6 -0
  333. package/hedhog/table/question.yaml +10 -0
  334. package/hedhog/table/question_subject.yaml +17 -0
  335. package/hedhog/table/student_activity_streak.yaml +25 -0
  336. package/package.json +6 -6
  337. package/src/achievement/achievement.controller.ts +60 -0
  338. package/src/achievement/achievement.mcp-tools.ts +108 -0
  339. package/src/achievement/achievement.module.ts +13 -0
  340. package/src/achievement/achievement.service.ts +252 -0
  341. package/src/achievement/dto/create-achievement.dto.ts +50 -0
  342. package/src/achievement/dto/update-achievement.dto.ts +47 -0
  343. package/src/bitcode-wallet/bitcode-wallet.controller.ts +69 -0
  344. package/src/bitcode-wallet/bitcode-wallet.mcp-tools.ts +107 -0
  345. package/src/bitcode-wallet/bitcode-wallet.module.ts +13 -0
  346. package/src/bitcode-wallet/bitcode-wallet.service.ts +361 -0
  347. package/src/bitcode-wallet/dto/create-bitcode-wallet-transaction.dto.ts +27 -0
  348. package/src/bitcode-wallet/dto/create-bitcode-wallet.dto.ts +7 -0
  349. package/src/bitcode-wallet/dto/update-bitcode-wallet-transaction.dto.ts +28 -0
  350. package/src/bitcode-wallet/dto/update-bitcode-wallet.dto.ts +8 -0
  351. package/src/certificate/certificate.controller.ts +20 -11
  352. package/src/certificate/certificate.mcp-tools.ts +131 -0
  353. package/src/certificate/certificate.module.ts +2 -1
  354. package/src/certificate/certificate.service.ts +95 -4
  355. package/src/certificate/dto/update-certificate-public-access.dto.ts +6 -0
  356. package/src/class-group/class-group.mcp-tools.ts +435 -0
  357. package/src/class-group/class-group.module.ts +2 -1
  358. package/src/class-group/class-group.service.ts +51 -1
  359. package/src/course/course-operations-integration.service.ts +520 -0
  360. package/src/course/course-structure.controller.ts +22 -8
  361. package/src/course/course-structure.service.ts +215 -23
  362. package/src/course/course.mcp-tools.ts +409 -0
  363. package/src/course/course.module.ts +8 -1
  364. package/src/course/course.service.ts +106 -27
  365. package/src/course/dto/create-course.dto.ts +8 -0
  366. package/src/course/dto/update-course-resources.dto.ts +39 -0
  367. package/src/course-lesson-discussion/course-lesson-discussion.controller.ts +55 -0
  368. package/src/course-lesson-discussion/course-lesson-discussion.mcp-tools.ts +75 -0
  369. package/src/course-lesson-discussion/course-lesson-discussion.module.ts +13 -0
  370. package/src/course-lesson-discussion/course-lesson-discussion.service.ts +354 -0
  371. package/src/course-lesson-discussion/dto/create-course-lesson-discussion-topic.dto.ts +16 -0
  372. package/src/course-lesson-note/course-lesson-note.controller.ts +68 -0
  373. package/src/course-lesson-note/course-lesson-note.mcp-tools.ts +96 -0
  374. package/src/course-lesson-note/course-lesson-note.module.ts +13 -0
  375. package/src/course-lesson-note/course-lesson-note.service.ts +248 -0
  376. package/src/course-lesson-note/dto/create-course-lesson-note.dto.ts +16 -0
  377. package/src/course-lesson-note/dto/update-course-lesson-note.dto.ts +18 -0
  378. package/src/dashboard/dashboard.mcp-tools.ts +23 -0
  379. package/src/dashboard/dashboard.module.ts +2 -1
  380. package/src/enterprise/enterprise.mcp-tools.ts +403 -0
  381. package/src/enterprise/enterprise.module.ts +2 -1
  382. package/src/enterprise/training/enterprise-training.module.ts +11 -1
  383. package/src/enterprise/training/training-admin.mcp-tools.ts +479 -0
  384. package/src/enterprise/training/training-instructor.mcp-tools.ts +210 -0
  385. package/src/enterprise/training/training-student.controller.ts +17 -1
  386. package/src/enterprise/training/training-student.mcp-tools.ts +136 -0
  387. package/src/enterprise/training/training-student.service.ts +167 -1
  388. package/src/evaluation/evaluation.mcp-tools.ts +155 -0
  389. package/src/evaluation/evaluation.module.ts +2 -1
  390. package/src/exam/dto/create-exam-question.dto.ts +8 -0
  391. package/src/exam/dto/create-exam.dto.ts +8 -0
  392. package/src/exam/dto/create-question-subject.dto.ts +12 -0
  393. package/src/exam/exam-attempt.service.ts +46 -14
  394. package/src/exam/exam.controller.ts +19 -0
  395. package/src/exam/exam.mcp-tools.ts +337 -0
  396. package/src/exam/exam.module.ts +2 -1
  397. package/src/exam/exam.service.ts +121 -0
  398. package/src/index.ts +9 -0
  399. package/src/instructor/instructor.mcp-tools.ts +243 -0
  400. package/src/instructor/instructor.module.ts +2 -1
  401. package/src/lms.module.ts +15 -1
  402. package/src/realtime/lms-realtime.controller.ts +12 -0
  403. package/src/realtime/lms-realtime.module.ts +12 -0
  404. package/src/realtime/lms-realtime.service.ts +98 -0
  405. package/src/realtime/lms-realtime.subscriber.ts +61 -0
  406. package/src/reports/reports.mcp-tools.ts +27 -0
  407. package/src/reports/reports.module.ts +2 -1
  408. package/src/training/training.mcp-tools.ts +128 -0
  409. package/src/training/training.module.ts +2 -1
@@ -0,0 +1,354 @@
1
+ import { PrismaService } from '@hed-hog/api-prisma';
2
+ import {
3
+ ForbiddenException,
4
+ Injectable,
5
+ NotFoundException,
6
+ } from '@nestjs/common';
7
+ import { CreateCourseLessonDiscussionTopicDto } from './dto/create-course-lesson-discussion-topic.dto';
8
+
9
+ type DiscussionRow = {
10
+ id: number;
11
+ lessonId: number;
12
+ personId: number;
13
+ parentTopicId: number | null;
14
+ content: string;
15
+ playerTimeSeconds: number;
16
+ likesCount: number;
17
+ likedByMe: boolean;
18
+ createdAt: string | null;
19
+ updatedAt: string | null;
20
+ authorName: string | null;
21
+ };
22
+
23
+ export type DiscussionNode = {
24
+ id: number;
25
+ lessonId: number;
26
+ personId: number;
27
+ parentTopicId: number | null;
28
+ content: string;
29
+ playerTimeSeconds: number;
30
+ likesCount: number;
31
+ likedByMe: boolean;
32
+ createdAt: string | null;
33
+ updatedAt: string | null;
34
+ author: {
35
+ personId: number;
36
+ name: string | null;
37
+ };
38
+ replies: DiscussionNode[];
39
+ };
40
+
41
+ @Injectable()
42
+ export class CourseLessonDiscussionService {
43
+ constructor(private readonly prisma: PrismaService) {}
44
+
45
+ async listTopics(lessonId: number, userId: number) {
46
+ const personId = await this.assertStudentAccess(lessonId, userId);
47
+
48
+ const rows = await this.queryRows<DiscussionRow>(
49
+ `SELECT
50
+ t.id,
51
+ t.course_lesson_id AS "lessonId",
52
+ t.person_id AS "personId",
53
+ t.parent_topic_id AS "parentTopicId",
54
+ t.content,
55
+ t.player_time_seconds AS "playerTimeSeconds",
56
+ t.created_at::text AS "createdAt",
57
+ t.updated_at::text AS "updatedAt",
58
+ p.name AS "authorName",
59
+ COALESCE(
60
+ (
61
+ SELECT COUNT(*)
62
+ FROM course_lesson_discussion_like l
63
+ WHERE l.topic_id = t.id
64
+ ),
65
+ 0
66
+ )::int AS "likesCount",
67
+ EXISTS(
68
+ SELECT 1
69
+ FROM course_lesson_discussion_like l
70
+ WHERE l.topic_id = t.id
71
+ AND l.person_id = $2
72
+ ) AS "likedByMe"
73
+ FROM course_lesson_discussion_topic t
74
+ JOIN person p ON p.id = t.person_id
75
+ WHERE t.course_lesson_id = $1
76
+ ORDER BY t.created_at ASC, t.id ASC`,
77
+ [lessonId, personId]
78
+ );
79
+
80
+ const tree = this.buildTree(rows.map((row) => this.mapRow(row)));
81
+
82
+ return {
83
+ lessonId,
84
+ totalTopics: rows.length,
85
+ topics: tree,
86
+ };
87
+ }
88
+
89
+ async createTopic(
90
+ lessonId: number,
91
+ userId: number,
92
+ dto: CreateCourseLessonDiscussionTopicDto
93
+ ) {
94
+ const personId = await this.assertStudentAccess(lessonId, userId);
95
+
96
+ if (dto.parentTopicId) {
97
+ const parent = await this.querySingle<{ id: number }>(
98
+ `SELECT id
99
+ FROM course_lesson_discussion_topic
100
+ WHERE id = $1
101
+ AND course_lesson_id = $2`,
102
+ [dto.parentTopicId, lessonId]
103
+ );
104
+
105
+ if (!parent) {
106
+ throw new NotFoundException('Parent topic not found');
107
+ }
108
+ }
109
+
110
+ const created = await this.querySingle<{ id: number }>(
111
+ `INSERT INTO course_lesson_discussion_topic (
112
+ course_lesson_id,
113
+ person_id,
114
+ parent_topic_id,
115
+ content,
116
+ player_time_seconds,
117
+ created_at,
118
+ updated_at
119
+ ) VALUES (
120
+ $1,
121
+ $2,
122
+ $3,
123
+ $4,
124
+ $5,
125
+ NOW(),
126
+ NOW()
127
+ )
128
+ RETURNING id`,
129
+ [
130
+ lessonId,
131
+ personId,
132
+ dto.parentTopicId ?? null,
133
+ this.sanitizeHtml(dto.content),
134
+ Number(dto.playerTimeSeconds),
135
+ ]
136
+ );
137
+
138
+ return this.getTopicById(lessonId, created.id, personId);
139
+ }
140
+
141
+ async likeTopic(lessonId: number, topicId: number, userId: number) {
142
+ const personId = await this.assertStudentAccess(lessonId, userId);
143
+ await this.assertTopicBelongsToLesson(lessonId, topicId);
144
+
145
+ await this.prisma.$executeRawUnsafe(
146
+ `INSERT INTO course_lesson_discussion_like (
147
+ topic_id,
148
+ person_id,
149
+ created_at
150
+ ) VALUES (
151
+ $1,
152
+ $2,
153
+ NOW()
154
+ )
155
+ ON CONFLICT (topic_id, person_id) DO NOTHING`,
156
+ topicId,
157
+ personId
158
+ );
159
+
160
+ return this.getLikeStatus(topicId, personId);
161
+ }
162
+
163
+ async unlikeTopic(lessonId: number, topicId: number, userId: number) {
164
+ const personId = await this.assertStudentAccess(lessonId, userId);
165
+ await this.assertTopicBelongsToLesson(lessonId, topicId);
166
+
167
+ await this.prisma.$executeRawUnsafe(
168
+ `DELETE FROM course_lesson_discussion_like
169
+ WHERE topic_id = $1
170
+ AND person_id = $2`,
171
+ topicId,
172
+ personId
173
+ );
174
+
175
+ return this.getLikeStatus(topicId, personId);
176
+ }
177
+
178
+ private async getTopicById(lessonId: number, topicId: number, viewerPersonId: number) {
179
+ const row = await this.querySingle<DiscussionRow>(
180
+ `SELECT
181
+ t.id,
182
+ t.course_lesson_id AS "lessonId",
183
+ t.person_id AS "personId",
184
+ t.parent_topic_id AS "parentTopicId",
185
+ t.content,
186
+ t.player_time_seconds AS "playerTimeSeconds",
187
+ t.created_at::text AS "createdAt",
188
+ t.updated_at::text AS "updatedAt",
189
+ p.name AS "authorName",
190
+ COALESCE(
191
+ (
192
+ SELECT COUNT(*)
193
+ FROM course_lesson_discussion_like l
194
+ WHERE l.topic_id = t.id
195
+ ),
196
+ 0
197
+ )::int AS "likesCount",
198
+ EXISTS(
199
+ SELECT 1
200
+ FROM course_lesson_discussion_like l
201
+ WHERE l.topic_id = t.id
202
+ AND l.person_id = $2
203
+ ) AS "likedByMe"
204
+ FROM course_lesson_discussion_topic t
205
+ JOIN person p ON p.id = t.person_id
206
+ WHERE t.id = $1
207
+ AND t.course_lesson_id = $3`,
208
+ [topicId, viewerPersonId, lessonId]
209
+ );
210
+
211
+ if (!row) {
212
+ throw new NotFoundException('Topic not found');
213
+ }
214
+
215
+ return this.mapRow(row);
216
+ }
217
+
218
+ private async getLikeStatus(topicId: number, personId: number) {
219
+ const status = await this.querySingle<{ likesCount: number; likedByMe: boolean }>(
220
+ `SELECT
221
+ COALESCE((SELECT COUNT(*) FROM course_lesson_discussion_like WHERE topic_id = $1), 0)::int AS "likesCount",
222
+ EXISTS(
223
+ SELECT 1
224
+ FROM course_lesson_discussion_like
225
+ WHERE topic_id = $1
226
+ AND person_id = $2
227
+ ) AS "likedByMe"`,
228
+ [topicId, personId]
229
+ );
230
+
231
+ return {
232
+ topicId,
233
+ likesCount: Number(status?.likesCount ?? 0),
234
+ likedByMe: Boolean(status?.likedByMe),
235
+ };
236
+ }
237
+
238
+ private buildTree(nodes: DiscussionNode[]) {
239
+ const byId = new Map<number, DiscussionNode>();
240
+ const roots: DiscussionNode[] = [];
241
+
242
+ for (const node of nodes) {
243
+ byId.set(node.id, node);
244
+ }
245
+
246
+ for (const node of nodes) {
247
+ if (node.parentTopicId && byId.has(node.parentTopicId)) {
248
+ byId.get(node.parentTopicId)?.replies.push(node);
249
+ } else {
250
+ roots.push(node);
251
+ }
252
+ }
253
+
254
+ return roots;
255
+ }
256
+
257
+ private mapRow(row: DiscussionRow): DiscussionNode {
258
+ return {
259
+ id: row.id,
260
+ lessonId: row.lessonId,
261
+ personId: row.personId,
262
+ parentTopicId: row.parentTopicId,
263
+ content: row.content,
264
+ playerTimeSeconds: Number(row.playerTimeSeconds ?? 0),
265
+ likesCount: Number(row.likesCount ?? 0),
266
+ likedByMe: Boolean(row.likedByMe),
267
+ createdAt: row.createdAt,
268
+ updatedAt: row.updatedAt,
269
+ author: {
270
+ personId: row.personId,
271
+ name: row.authorName,
272
+ },
273
+ replies: [],
274
+ };
275
+ }
276
+
277
+ private sanitizeHtml(content: string) {
278
+ const trimmed = content.trim();
279
+
280
+ const noScripts = trimmed
281
+ .replace(/<script[\s\S]*?>[\s\S]*?<\/script>/gi, '')
282
+ .replace(/<style[\s\S]*?>[\s\S]*?<\/style>/gi, '');
283
+
284
+ const noInlineEvents = noScripts.replace(/\son\w+=("[^"]*"|'[^']*'|[^\s>]+)/gi, '');
285
+ const safeLinks = noInlineEvents.replace(/javascript:/gi, '');
286
+
287
+ return safeLinks;
288
+ }
289
+
290
+ private async assertTopicBelongsToLesson(lessonId: number, topicId: number) {
291
+ const topic = await this.querySingle<{ id: number }>(
292
+ `SELECT id
293
+ FROM course_lesson_discussion_topic
294
+ WHERE id = $1
295
+ AND course_lesson_id = $2`,
296
+ [topicId, lessonId]
297
+ );
298
+
299
+ if (!topic) {
300
+ throw new NotFoundException('Topic not found');
301
+ }
302
+ }
303
+
304
+ private async assertStudentAccess(lessonId: number, userId: number) {
305
+ const personUser = await this.prisma.person_user.findFirst({
306
+ where: { user_id: userId },
307
+ select: { person_id: true },
308
+ });
309
+
310
+ if (!personUser) {
311
+ throw new NotFoundException('No person profile found for this user');
312
+ }
313
+
314
+ const lesson = await this.prisma.course_lesson.findUnique({
315
+ where: { id: lessonId },
316
+ select: {
317
+ id: true,
318
+ course_module: {
319
+ select: {
320
+ course_id: true,
321
+ },
322
+ },
323
+ },
324
+ });
325
+
326
+ if (!lesson?.course_module?.course_id) {
327
+ throw new NotFoundException('Lesson not found');
328
+ }
329
+
330
+ const enrollment = await this.prisma.course_enrollment.findFirst({
331
+ where: {
332
+ person_id: personUser.person_id,
333
+ course_id: lesson.course_module.course_id,
334
+ status: { not: 'cancelled' },
335
+ },
336
+ select: { id: true },
337
+ });
338
+
339
+ if (!enrollment) {
340
+ throw new ForbiddenException('You do not have access to this lesson');
341
+ }
342
+
343
+ return personUser.person_id;
344
+ }
345
+
346
+ private async queryRows<T>(query: string, values: unknown[] = []) {
347
+ return this.prisma.$queryRawUnsafe<T[]>(query, ...values);
348
+ }
349
+
350
+ private async querySingle<T>(query: string, values: unknown[] = []) {
351
+ const rows = await this.prisma.$queryRawUnsafe<T[]>(query, ...values);
352
+ return rows[0] ?? null;
353
+ }
354
+ }
@@ -0,0 +1,16 @@
1
+ import { IsInt, IsOptional, IsString, MaxLength, Min } from 'class-validator';
2
+
3
+ export class CreateCourseLessonDiscussionTopicDto {
4
+ @IsString()
5
+ @MaxLength(20000)
6
+ content: string;
7
+
8
+ @IsInt()
9
+ @Min(0)
10
+ playerTimeSeconds: number;
11
+
12
+ @IsOptional()
13
+ @IsInt()
14
+ @Min(1)
15
+ parentTopicId?: number;
16
+ }
@@ -0,0 +1,68 @@
1
+ import { Role, User } from '@hed-hog/api';
2
+ import {
3
+ Body,
4
+ Controller,
5
+ Delete,
6
+ Get,
7
+ HttpCode,
8
+ HttpStatus,
9
+ Param,
10
+ ParseIntPipe,
11
+ Patch,
12
+ Post,
13
+ } from '@nestjs/common';
14
+ import { CourseLessonNoteService } from './course-lesson-note.service';
15
+ import { CreateCourseLessonNoteDto } from './dto/create-course-lesson-note.dto';
16
+ import { UpdateCourseLessonNoteDto } from './dto/update-course-lesson-note.dto';
17
+
18
+ @Role()
19
+ @Controller('lms/course-lessons/:lessonId/notes')
20
+ export class CourseLessonNoteController {
21
+ constructor(private readonly noteService: CourseLessonNoteService) {}
22
+
23
+ @Get()
24
+ list(
25
+ @Param('lessonId', ParseIntPipe) lessonId: number,
26
+ @User('id') userId: number
27
+ ) {
28
+ return this.noteService.list(lessonId, userId);
29
+ }
30
+
31
+ @Get(':id')
32
+ getById(
33
+ @Param('lessonId', ParseIntPipe) lessonId: number,
34
+ @Param('id', ParseIntPipe) id: number,
35
+ @User('id') userId: number
36
+ ) {
37
+ return this.noteService.getById(lessonId, id, userId);
38
+ }
39
+
40
+ @Post()
41
+ create(
42
+ @Param('lessonId', ParseIntPipe) lessonId: number,
43
+ @User('id') userId: number,
44
+ @Body() dto: CreateCourseLessonNoteDto
45
+ ) {
46
+ return this.noteService.create(lessonId, userId, dto);
47
+ }
48
+
49
+ @Patch(':id')
50
+ update(
51
+ @Param('lessonId', ParseIntPipe) lessonId: number,
52
+ @Param('id', ParseIntPipe) id: number,
53
+ @User('id') userId: number,
54
+ @Body() dto: UpdateCourseLessonNoteDto
55
+ ) {
56
+ return this.noteService.update(lessonId, id, userId, dto);
57
+ }
58
+
59
+ @Delete(':id')
60
+ @HttpCode(HttpStatus.NO_CONTENT)
61
+ remove(
62
+ @Param('lessonId', ParseIntPipe) lessonId: number,
63
+ @Param('id', ParseIntPipe) id: number,
64
+ @User('id') userId: number
65
+ ) {
66
+ return this.noteService.delete(lessonId, id, userId);
67
+ }
68
+ }
@@ -0,0 +1,96 @@
1
+ import { McpContext, McpTool } from '@hed-hog/core';
2
+ import { Injectable } from '@nestjs/common';
3
+ import { CourseLessonNoteService } from './course-lesson-note.service';
4
+
5
+ @Injectable()
6
+ export class LmsCourseLessonNoteMcpTools {
7
+ constructor(private readonly noteService: CourseLessonNoteService) {}
8
+
9
+ @McpTool({
10
+ name: 'lms.lesson-notes.list',
11
+ description: "Lists the authenticated user's notes for a course lesson, ordered by player timestamp.",
12
+ inputSchema: {
13
+ type: 'object',
14
+ properties: {
15
+ lessonId: { type: 'number', description: 'Course lesson ID' },
16
+ },
17
+ required: ['lessonId'],
18
+ },
19
+ readOnly: true,
20
+ })
21
+ async listNotes(args: { lessonId: number }, context: McpContext): Promise<any> {
22
+ return this.noteService.list(args.lessonId, context.userId);
23
+ }
24
+
25
+ @McpTool({
26
+ name: 'lms.lesson-notes.get',
27
+ description: "Returns a single note by ID for the authenticated user.",
28
+ inputSchema: {
29
+ type: 'object',
30
+ properties: {
31
+ lessonId: { type: 'number', description: 'Course lesson ID' },
32
+ id: { type: 'number', description: 'Note ID' },
33
+ },
34
+ required: ['lessonId', 'id'],
35
+ },
36
+ readOnly: true,
37
+ })
38
+ async getNote(args: { lessonId: number; id: number }, context: McpContext): Promise<any> {
39
+ return this.noteService.getById(args.lessonId, args.id, context.userId);
40
+ }
41
+
42
+ @McpTool({
43
+ name: 'lms.lesson-notes.create',
44
+ description: 'Creates a new note on a course lesson for the authenticated user.',
45
+ inputSchema: {
46
+ type: 'object',
47
+ properties: {
48
+ lessonId: { type: 'number', description: 'Course lesson ID' },
49
+ description: { type: 'string', description: 'Note text content (max 5000 chars)' },
50
+ playerTimeSeconds: { type: 'number', description: 'Video player timestamp in seconds' },
51
+ frameFileId: { type: 'number', description: 'Screenshot frame file ID (optional)' },
52
+ },
53
+ required: ['lessonId', 'description', 'playerTimeSeconds'],
54
+ },
55
+ })
56
+ async createNote(args: { lessonId: number; [key: string]: any }, context: McpContext): Promise<any> {
57
+ const { lessonId, ...dto } = args;
58
+ return this.noteService.create(lessonId, context.userId, dto as any);
59
+ }
60
+
61
+ @McpTool({
62
+ name: 'lms.lesson-notes.update',
63
+ description: "Updates one of the authenticated user's lesson notes.",
64
+ inputSchema: {
65
+ type: 'object',
66
+ properties: {
67
+ lessonId: { type: 'number', description: 'Course lesson ID' },
68
+ id: { type: 'number', description: 'Note ID' },
69
+ description: { type: 'string', description: 'Note text content' },
70
+ playerTimeSeconds: { type: 'number', description: 'Video player timestamp in seconds' },
71
+ frameFileId: { type: 'number', description: 'Screenshot frame file ID' },
72
+ },
73
+ required: ['lessonId', 'id'],
74
+ },
75
+ })
76
+ async updateNote(args: { lessonId: number; id: number; [key: string]: any }, context: McpContext): Promise<any> {
77
+ const { lessonId, id, ...dto } = args;
78
+ return this.noteService.update(lessonId, id, context.userId, dto as any);
79
+ }
80
+
81
+ @McpTool({
82
+ name: 'lms.lesson-notes.delete',
83
+ description: "Deletes one of the authenticated user's lesson notes.",
84
+ inputSchema: {
85
+ type: 'object',
86
+ properties: {
87
+ lessonId: { type: 'number', description: 'Course lesson ID' },
88
+ id: { type: 'number', description: 'Note ID' },
89
+ },
90
+ required: ['lessonId', 'id'],
91
+ },
92
+ })
93
+ async deleteNote(args: { lessonId: number; id: number }, context: McpContext): Promise<any> {
94
+ return this.noteService.delete(args.lessonId, args.id, context.userId);
95
+ }
96
+ }
@@ -0,0 +1,13 @@
1
+ import { PrismaModule } from '@hed-hog/api-prisma';
2
+ import { forwardRef, Module } from '@nestjs/common';
3
+ import { CourseLessonNoteController } from './course-lesson-note.controller';
4
+ import { LmsCourseLessonNoteMcpTools } from './course-lesson-note.mcp-tools';
5
+ import { CourseLessonNoteService } from './course-lesson-note.service';
6
+
7
+ @Module({
8
+ imports: [forwardRef(() => PrismaModule)],
9
+ controllers: [CourseLessonNoteController],
10
+ providers: [CourseLessonNoteService, LmsCourseLessonNoteMcpTools],
11
+ exports: [CourseLessonNoteService],
12
+ })
13
+ export class CourseLessonNoteModule {}