@hed-hog/lms 0.0.347 → 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 (414) 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.controller.d.ts +3 -3
  77. package/dist/class-group/class-group.mcp-tools.d.ts +87 -0
  78. package/dist/class-group/class-group.mcp-tools.d.ts.map +1 -0
  79. package/dist/class-group/class-group.mcp-tools.js +553 -0
  80. package/dist/class-group/class-group.mcp-tools.js.map +1 -0
  81. package/dist/class-group/class-group.module.d.ts.map +1 -1
  82. package/dist/class-group/class-group.module.js +2 -1
  83. package/dist/class-group/class-group.module.js.map +1 -1
  84. package/dist/class-group/class-group.service.d.ts +6 -4
  85. package/dist/class-group/class-group.service.d.ts.map +1 -1
  86. package/dist/class-group/class-group.service.js +45 -2
  87. package/dist/class-group/class-group.service.js.map +1 -1
  88. package/dist/course/course-operations-integration.service.d.ts +40 -0
  89. package/dist/course/course-operations-integration.service.d.ts.map +1 -0
  90. package/dist/course/course-operations-integration.service.js +372 -0
  91. package/dist/course/course-operations-integration.service.js.map +1 -0
  92. package/dist/course/course-structure.controller.d.ts +43 -4
  93. package/dist/course/course-structure.controller.d.ts.map +1 -1
  94. package/dist/course/course-structure.controller.js +22 -0
  95. package/dist/course/course-structure.controller.js.map +1 -1
  96. package/dist/course/course-structure.service.d.ts +42 -1
  97. package/dist/course/course-structure.service.d.ts.map +1 -1
  98. package/dist/course/course-structure.service.js +199 -32
  99. package/dist/course/course-structure.service.js.map +1 -1
  100. package/dist/course/course.controller.d.ts +12 -0
  101. package/dist/course/course.controller.d.ts.map +1 -1
  102. package/dist/course/course.mcp-tools.d.ts +90 -0
  103. package/dist/course/course.mcp-tools.d.ts.map +1 -0
  104. package/dist/course/course.mcp-tools.js +520 -0
  105. package/dist/course/course.mcp-tools.js.map +1 -0
  106. package/dist/course/course.module.d.ts.map +1 -1
  107. package/dist/course/course.module.js +8 -1
  108. package/dist/course/course.module.js.map +1 -1
  109. package/dist/course/course.service.d.ts +15 -1
  110. package/dist/course/course.service.d.ts.map +1 -1
  111. package/dist/course/course.service.js +70 -35
  112. package/dist/course/course.service.js.map +1 -1
  113. package/dist/course/dto/create-course.dto.d.ts +1 -0
  114. package/dist/course/dto/create-course.dto.d.ts.map +1 -1
  115. package/dist/course/dto/create-course.dto.js +7 -0
  116. package/dist/course/dto/create-course.dto.js.map +1 -1
  117. package/dist/course/dto/update-course-resources.dto.d.ts +11 -0
  118. package/dist/course/dto/update-course-resources.dto.d.ts.map +1 -0
  119. package/dist/course/dto/update-course-resources.dto.js +51 -0
  120. package/dist/course/dto/update-course-resources.dto.js.map +1 -0
  121. package/dist/course-lesson-discussion/course-lesson-discussion.controller.d.ts +23 -0
  122. package/dist/course-lesson-discussion/course-lesson-discussion.controller.d.ts.map +1 -0
  123. package/dist/course-lesson-discussion/course-lesson-discussion.controller.js +78 -0
  124. package/dist/course-lesson-discussion/course-lesson-discussion.controller.js.map +1 -0
  125. package/dist/course-lesson-discussion/course-lesson-discussion.mcp-tools.d.ts +22 -0
  126. package/dist/course-lesson-discussion/course-lesson-discussion.mcp-tools.d.ts.map +1 -0
  127. package/dist/course-lesson-discussion/course-lesson-discussion.mcp-tools.js +120 -0
  128. package/dist/course-lesson-discussion/course-lesson-discussion.mcp-tools.js.map +1 -0
  129. package/dist/course-lesson-discussion/course-lesson-discussion.module.d.ts +3 -0
  130. package/dist/course-lesson-discussion/course-lesson-discussion.module.d.ts.map +1 -0
  131. package/dist/course-lesson-discussion/course-lesson-discussion.module.js +26 -0
  132. package/dist/course-lesson-discussion/course-lesson-discussion.module.js.map +1 -0
  133. package/dist/course-lesson-discussion/course-lesson-discussion.service.d.ts +49 -0
  134. package/dist/course-lesson-discussion/course-lesson-discussion.service.d.ts.map +1 -0
  135. package/dist/course-lesson-discussion/course-lesson-discussion.service.js +272 -0
  136. package/dist/course-lesson-discussion/course-lesson-discussion.service.js.map +1 -0
  137. package/dist/course-lesson-discussion/dto/create-course-lesson-discussion-topic.dto.d.ts +6 -0
  138. package/dist/course-lesson-discussion/dto/create-course-lesson-discussion-topic.dto.d.ts.map +1 -0
  139. package/dist/course-lesson-discussion/dto/create-course-lesson-discussion-topic.dto.js +33 -0
  140. package/dist/course-lesson-discussion/dto/create-course-lesson-discussion-topic.dto.js.map +1 -0
  141. package/dist/course-lesson-note/course-lesson-note.controller.d.ts +53 -0
  142. package/dist/course-lesson-note/course-lesson-note.controller.d.ts.map +1 -0
  143. package/dist/course-lesson-note/course-lesson-note.controller.js +93 -0
  144. package/dist/course-lesson-note/course-lesson-note.controller.js.map +1 -0
  145. package/dist/course-lesson-note/course-lesson-note.mcp-tools.d.ts +27 -0
  146. package/dist/course-lesson-note/course-lesson-note.mcp-tools.d.ts.map +1 -0
  147. package/dist/course-lesson-note/course-lesson-note.mcp-tools.js +145 -0
  148. package/dist/course-lesson-note/course-lesson-note.mcp-tools.js.map +1 -0
  149. package/dist/course-lesson-note/course-lesson-note.module.d.ts +3 -0
  150. package/dist/course-lesson-note/course-lesson-note.module.d.ts.map +1 -0
  151. package/dist/course-lesson-note/course-lesson-note.module.js +26 -0
  152. package/dist/course-lesson-note/course-lesson-note.module.js.map +1 -0
  153. package/dist/course-lesson-note/course-lesson-note.service.d.ts +59 -0
  154. package/dist/course-lesson-note/course-lesson-note.service.d.ts.map +1 -0
  155. package/dist/course-lesson-note/course-lesson-note.service.js +195 -0
  156. package/dist/course-lesson-note/course-lesson-note.service.js.map +1 -0
  157. package/dist/course-lesson-note/dto/create-course-lesson-note.dto.d.ts +6 -0
  158. package/dist/course-lesson-note/dto/create-course-lesson-note.dto.d.ts.map +1 -0
  159. package/dist/course-lesson-note/dto/create-course-lesson-note.dto.js +33 -0
  160. package/dist/course-lesson-note/dto/create-course-lesson-note.dto.js.map +1 -0
  161. package/dist/course-lesson-note/dto/update-course-lesson-note.dto.d.ts +6 -0
  162. package/dist/course-lesson-note/dto/update-course-lesson-note.dto.d.ts.map +1 -0
  163. package/dist/course-lesson-note/dto/update-course-lesson-note.dto.js +35 -0
  164. package/dist/course-lesson-note/dto/update-course-lesson-note.dto.js.map +1 -0
  165. package/dist/dashboard/dashboard.mcp-tools.d.ts +10 -0
  166. package/dist/dashboard/dashboard.mcp-tools.d.ts.map +1 -0
  167. package/dist/dashboard/dashboard.mcp-tools.js +46 -0
  168. package/dist/dashboard/dashboard.mcp-tools.js.map +1 -0
  169. package/dist/dashboard/dashboard.module.d.ts.map +1 -1
  170. package/dist/dashboard/dashboard.module.js +2 -1
  171. package/dist/dashboard/dashboard.module.js.map +1 -1
  172. package/dist/enterprise/enterprise.mcp-tools.d.ts +82 -0
  173. package/dist/enterprise/enterprise.mcp-tools.d.ts.map +1 -0
  174. package/dist/enterprise/enterprise.mcp-tools.js +516 -0
  175. package/dist/enterprise/enterprise.mcp-tools.js.map +1 -0
  176. package/dist/enterprise/enterprise.module.d.ts.map +1 -1
  177. package/dist/enterprise/enterprise.module.js +2 -1
  178. package/dist/enterprise/enterprise.module.js.map +1 -1
  179. package/dist/enterprise/training/enterprise-training.module.d.ts.map +1 -1
  180. package/dist/enterprise/training/enterprise-training.module.js +11 -1
  181. package/dist/enterprise/training/enterprise-training.module.js.map +1 -1
  182. package/dist/enterprise/training/training-admin.controller.d.ts +2 -2
  183. package/dist/enterprise/training/training-admin.mcp-tools.d.ts +79 -0
  184. package/dist/enterprise/training/training-admin.mcp-tools.d.ts.map +1 -0
  185. package/dist/enterprise/training/training-admin.mcp-tools.js +620 -0
  186. package/dist/enterprise/training/training-admin.mcp-tools.js.map +1 -0
  187. package/dist/enterprise/training/training-admin.service.d.ts +2 -2
  188. package/dist/enterprise/training/training-instructor.mcp-tools.d.ts +47 -0
  189. package/dist/enterprise/training/training-instructor.mcp-tools.d.ts.map +1 -0
  190. package/dist/enterprise/training/training-instructor.mcp-tools.js +275 -0
  191. package/dist/enterprise/training/training-instructor.mcp-tools.js.map +1 -0
  192. package/dist/enterprise/training/training-student.controller.d.ts +24 -0
  193. package/dist/enterprise/training/training-student.controller.d.ts.map +1 -1
  194. package/dist/enterprise/training/training-student.controller.js +22 -0
  195. package/dist/enterprise/training/training-student.controller.js.map +1 -1
  196. package/dist/enterprise/training/training-student.mcp-tools.d.ts +27 -0
  197. package/dist/enterprise/training/training-student.mcp-tools.d.ts.map +1 -0
  198. package/dist/enterprise/training/training-student.mcp-tools.js +186 -0
  199. package/dist/enterprise/training/training-student.mcp-tools.js.map +1 -0
  200. package/dist/enterprise/training/training-student.service.d.ts +32 -0
  201. package/dist/enterprise/training/training-student.service.d.ts.map +1 -1
  202. package/dist/enterprise/training/training-student.service.js +138 -0
  203. package/dist/enterprise/training/training-student.service.js.map +1 -1
  204. package/dist/evaluation/evaluation.controller.d.ts +4 -4
  205. package/dist/evaluation/evaluation.mcp-tools.d.ts +25 -0
  206. package/dist/evaluation/evaluation.mcp-tools.d.ts.map +1 -0
  207. package/dist/evaluation/evaluation.mcp-tools.js +220 -0
  208. package/dist/evaluation/evaluation.mcp-tools.js.map +1 -0
  209. package/dist/evaluation/evaluation.module.d.ts.map +1 -1
  210. package/dist/evaluation/evaluation.module.js +2 -1
  211. package/dist/evaluation/evaluation.module.js.map +1 -1
  212. package/dist/evaluation/evaluation.service.d.ts +4 -4
  213. package/dist/exam/dto/create-exam-question.dto.d.ts +2 -0
  214. package/dist/exam/dto/create-exam-question.dto.d.ts.map +1 -1
  215. package/dist/exam/dto/create-exam-question.dto.js +10 -0
  216. package/dist/exam/dto/create-exam-question.dto.js.map +1 -1
  217. package/dist/exam/dto/create-exam.dto.d.ts +2 -0
  218. package/dist/exam/dto/create-exam.dto.d.ts.map +1 -1
  219. package/dist/exam/dto/create-exam.dto.js +10 -0
  220. package/dist/exam/dto/create-exam.dto.js.map +1 -1
  221. package/dist/exam/dto/create-question-subject.dto.d.ts +5 -0
  222. package/dist/exam/dto/create-question-subject.dto.d.ts.map +1 -0
  223. package/dist/exam/dto/create-question-subject.dto.js +28 -0
  224. package/dist/exam/dto/create-question-subject.dto.js.map +1 -0
  225. package/dist/exam/exam-attempt.controller.d.ts +4 -0
  226. package/dist/exam/exam-attempt.controller.d.ts.map +1 -1
  227. package/dist/exam/exam-attempt.service.d.ts +7 -1
  228. package/dist/exam/exam-attempt.service.d.ts.map +1 -1
  229. package/dist/exam/exam-attempt.service.js +47 -17
  230. package/dist/exam/exam-attempt.service.js.map +1 -1
  231. package/dist/exam/exam.controller.d.ts +34 -0
  232. package/dist/exam/exam.controller.d.ts.map +1 -1
  233. package/dist/exam/exam.controller.js +27 -0
  234. package/dist/exam/exam.controller.js.map +1 -1
  235. package/dist/exam/exam.mcp-tools.d.ts +62 -0
  236. package/dist/exam/exam.mcp-tools.d.ts.map +1 -0
  237. package/dist/exam/exam.mcp-tools.js +430 -0
  238. package/dist/exam/exam.mcp-tools.js.map +1 -0
  239. package/dist/exam/exam.module.d.ts.map +1 -1
  240. package/dist/exam/exam.module.js +2 -1
  241. package/dist/exam/exam.module.js.map +1 -1
  242. package/dist/exam/exam.service.d.ts +38 -0
  243. package/dist/exam/exam.service.d.ts.map +1 -1
  244. package/dist/exam/exam.service.js +114 -17
  245. package/dist/exam/exam.service.js.map +1 -1
  246. package/dist/index.d.ts +9 -0
  247. package/dist/index.d.ts.map +1 -1
  248. package/dist/index.js +9 -0
  249. package/dist/index.js.map +1 -1
  250. package/dist/instructor/instructor.mcp-tools.d.ts +41 -0
  251. package/dist/instructor/instructor.mcp-tools.d.ts.map +1 -0
  252. package/dist/instructor/instructor.mcp-tools.js +326 -0
  253. package/dist/instructor/instructor.mcp-tools.js.map +1 -0
  254. package/dist/instructor/instructor.module.d.ts.map +1 -1
  255. package/dist/instructor/instructor.module.js +2 -1
  256. package/dist/instructor/instructor.module.js.map +1 -1
  257. package/dist/lms.module.d.ts.map +1 -1
  258. package/dist/lms.module.js +15 -0
  259. package/dist/lms.module.js.map +1 -1
  260. package/dist/realtime/lms-realtime.controller.d.ts +7 -0
  261. package/dist/realtime/lms-realtime.controller.d.ts.map +1 -0
  262. package/dist/realtime/lms-realtime.controller.js +34 -0
  263. package/dist/realtime/lms-realtime.controller.js.map +1 -0
  264. package/dist/realtime/lms-realtime.module.d.ts +3 -0
  265. package/dist/realtime/lms-realtime.module.d.ts.map +1 -0
  266. package/dist/realtime/lms-realtime.module.js +25 -0
  267. package/dist/realtime/lms-realtime.module.js.map +1 -0
  268. package/dist/realtime/lms-realtime.service.d.ts +36 -0
  269. package/dist/realtime/lms-realtime.service.d.ts.map +1 -0
  270. package/dist/realtime/lms-realtime.service.js +59 -0
  271. package/dist/realtime/lms-realtime.service.js.map +1 -0
  272. package/dist/realtime/lms-realtime.subscriber.d.ts +10 -0
  273. package/dist/realtime/lms-realtime.subscriber.d.ts.map +1 -0
  274. package/dist/realtime/lms-realtime.subscriber.js +70 -0
  275. package/dist/realtime/lms-realtime.subscriber.js.map +1 -0
  276. package/dist/reports/reports.mcp-tools.d.ts +10 -0
  277. package/dist/reports/reports.mcp-tools.d.ts.map +1 -0
  278. package/dist/reports/reports.mcp-tools.js +50 -0
  279. package/dist/reports/reports.mcp-tools.js.map +1 -0
  280. package/dist/reports/reports.module.d.ts.map +1 -1
  281. package/dist/reports/reports.module.js +2 -1
  282. package/dist/reports/reports.module.js.map +1 -1
  283. package/dist/training/training.mcp-tools.d.ts +20 -0
  284. package/dist/training/training.mcp-tools.d.ts.map +1 -0
  285. package/dist/training/training.mcp-tools.js +181 -0
  286. package/dist/training/training.mcp-tools.js.map +1 -0
  287. package/dist/training/training.module.d.ts.map +1 -1
  288. package/dist/training/training.module.js +2 -1
  289. package/dist/training/training.module.js.map +1 -1
  290. package/hedhog/data/integration_event_catalog.yaml +69 -0
  291. package/hedhog/data/menu.yaml +34 -0
  292. package/hedhog/data/route.yaml +2351 -0
  293. package/hedhog/frontend/app/_components/class-form-sheet.tsx.ejs +168 -103
  294. package/hedhog/frontend/app/_components/course-form-sheet.tsx.ejs +80 -1
  295. package/hedhog/frontend/app/_components/course-picker.tsx.ejs +228 -0
  296. package/hedhog/frontend/app/_lib/hooks/use-lms-realtime-refresh.ts.ejs +58 -0
  297. package/hedhog/frontend/app/achievements/page.tsx.ejs +844 -0
  298. package/hedhog/frontend/app/bitcodes/page.tsx.ejs +1010 -0
  299. package/hedhog/frontend/app/certificates/issued/page.tsx.ejs +61 -2
  300. package/hedhog/frontend/app/classes/[id]/page.tsx.ejs +7 -0
  301. package/hedhog/frontend/app/classes/page.tsx.ejs +55 -2060
  302. package/hedhog/frontend/app/courses/[id]/_components/CourseRelationsCard.tsx.ejs +28 -0
  303. package/hedhog/frontend/app/courses/[id]/_components/course-edit-types.ts.ejs +1 -0
  304. package/hedhog/frontend/app/courses/[id]/page.tsx.ejs +0 -9
  305. package/hedhog/frontend/app/courses/[id]/structure/_components/course-tree-dnd.tsx.ejs +80 -66
  306. package/hedhog/frontend/app/courses/[id]/structure/_components/editor-course.tsx.ejs +583 -8
  307. package/hedhog/frontend/app/courses/[id]/structure/_components/editor-lesson.tsx.ejs +527 -57
  308. package/hedhog/frontend/app/courses/[id]/structure/_components/mock-data.ts.ejs +1 -1
  309. package/hedhog/frontend/app/courses/[id]/structure/_components/tree-context-menu.tsx.ejs +106 -4
  310. package/hedhog/frontend/app/courses/[id]/structure/_components/types.ts.ejs +2 -1
  311. package/hedhog/frontend/app/courses/[id]/structure/_data/adapters/course-structure.adapter.ts.ejs +11 -1
  312. package/hedhog/frontend/app/courses/[id]/structure/_data/services/course-structure.service.ts.ejs +53 -6
  313. package/hedhog/frontend/app/courses/[id]/structure/_data/types/api-course.types.ts.ejs +13 -2
  314. package/hedhog/frontend/app/courses/page.tsx.ejs +175 -29
  315. package/hedhog/frontend/app/enterprise/_components/enterprise-course-create-sheet.tsx.ejs +3 -0
  316. package/hedhog/frontend/app/enterprise/_components/enterprise-course-edit-sheet.tsx.ejs +7 -0
  317. package/hedhog/frontend/app/exams/[id]/questions/page.tsx.ejs +169 -2
  318. package/hedhog/frontend/app/exams/page.tsx.ejs +77 -22
  319. package/hedhog/frontend/app/instructor-skills/page.tsx.ejs +1 -0
  320. package/hedhog/frontend/app/instructors/page.tsx.ejs +1 -0
  321. package/hedhog/frontend/app/paths/page.tsx.ejs +6 -24
  322. package/hedhog/frontend/app/training/page.tsx.ejs +6 -24
  323. package/hedhog/frontend/messages/en.json +314 -12
  324. package/hedhog/frontend/messages/pt.json +314 -12
  325. package/hedhog/query/triggers.sql +53 -0
  326. package/hedhog/table/achievement.yaml +46 -0
  327. package/hedhog/table/bitcode_wallet.yaml +18 -0
  328. package/hedhog/table/bitcode_wallet_transaction.yaml +22 -0
  329. package/hedhog/table/certificate.yaml +3 -0
  330. package/hedhog/table/course.yaml +4 -0
  331. package/hedhog/table/course_file.yaml +23 -0
  332. package/hedhog/table/course_lesson.yaml +5 -0
  333. package/hedhog/table/course_lesson_discussion_like.yaml +21 -0
  334. package/hedhog/table/course_lesson_discussion_topic.yaml +35 -0
  335. package/hedhog/table/course_lesson_note.yaml +34 -0
  336. package/hedhog/table/exam.yaml +5 -0
  337. package/hedhog/table/learning_path_enrollment.yaml +6 -0
  338. package/hedhog/table/question.yaml +10 -0
  339. package/hedhog/table/question_subject.yaml +17 -0
  340. package/hedhog/table/student_activity_streak.yaml +25 -0
  341. package/package.json +7 -7
  342. package/src/achievement/achievement.controller.ts +60 -0
  343. package/src/achievement/achievement.mcp-tools.ts +108 -0
  344. package/src/achievement/achievement.module.ts +13 -0
  345. package/src/achievement/achievement.service.ts +252 -0
  346. package/src/achievement/dto/create-achievement.dto.ts +50 -0
  347. package/src/achievement/dto/update-achievement.dto.ts +47 -0
  348. package/src/bitcode-wallet/bitcode-wallet.controller.ts +69 -0
  349. package/src/bitcode-wallet/bitcode-wallet.mcp-tools.ts +107 -0
  350. package/src/bitcode-wallet/bitcode-wallet.module.ts +13 -0
  351. package/src/bitcode-wallet/bitcode-wallet.service.ts +361 -0
  352. package/src/bitcode-wallet/dto/create-bitcode-wallet-transaction.dto.ts +27 -0
  353. package/src/bitcode-wallet/dto/create-bitcode-wallet.dto.ts +7 -0
  354. package/src/bitcode-wallet/dto/update-bitcode-wallet-transaction.dto.ts +28 -0
  355. package/src/bitcode-wallet/dto/update-bitcode-wallet.dto.ts +8 -0
  356. package/src/certificate/certificate.controller.ts +20 -11
  357. package/src/certificate/certificate.mcp-tools.ts +131 -0
  358. package/src/certificate/certificate.module.ts +2 -1
  359. package/src/certificate/certificate.service.ts +95 -4
  360. package/src/certificate/dto/update-certificate-public-access.dto.ts +6 -0
  361. package/src/class-group/class-group.mcp-tools.ts +435 -0
  362. package/src/class-group/class-group.module.ts +2 -1
  363. package/src/class-group/class-group.service.ts +51 -1
  364. package/src/course/course-operations-integration.service.ts +520 -0
  365. package/src/course/course-structure.controller.ts +22 -8
  366. package/src/course/course-structure.service.ts +215 -23
  367. package/src/course/course.mcp-tools.ts +409 -0
  368. package/src/course/course.module.ts +8 -1
  369. package/src/course/course.service.ts +106 -27
  370. package/src/course/dto/create-course.dto.ts +8 -0
  371. package/src/course/dto/update-course-resources.dto.ts +39 -0
  372. package/src/course-lesson-discussion/course-lesson-discussion.controller.ts +55 -0
  373. package/src/course-lesson-discussion/course-lesson-discussion.mcp-tools.ts +75 -0
  374. package/src/course-lesson-discussion/course-lesson-discussion.module.ts +13 -0
  375. package/src/course-lesson-discussion/course-lesson-discussion.service.ts +354 -0
  376. package/src/course-lesson-discussion/dto/create-course-lesson-discussion-topic.dto.ts +16 -0
  377. package/src/course-lesson-note/course-lesson-note.controller.ts +68 -0
  378. package/src/course-lesson-note/course-lesson-note.mcp-tools.ts +96 -0
  379. package/src/course-lesson-note/course-lesson-note.module.ts +13 -0
  380. package/src/course-lesson-note/course-lesson-note.service.ts +248 -0
  381. package/src/course-lesson-note/dto/create-course-lesson-note.dto.ts +16 -0
  382. package/src/course-lesson-note/dto/update-course-lesson-note.dto.ts +18 -0
  383. package/src/dashboard/dashboard.mcp-tools.ts +23 -0
  384. package/src/dashboard/dashboard.module.ts +2 -1
  385. package/src/enterprise/enterprise.mcp-tools.ts +403 -0
  386. package/src/enterprise/enterprise.module.ts +2 -1
  387. package/src/enterprise/training/enterprise-training.module.ts +11 -1
  388. package/src/enterprise/training/training-admin.mcp-tools.ts +479 -0
  389. package/src/enterprise/training/training-instructor.mcp-tools.ts +210 -0
  390. package/src/enterprise/training/training-student.controller.ts +17 -1
  391. package/src/enterprise/training/training-student.mcp-tools.ts +136 -0
  392. package/src/enterprise/training/training-student.service.ts +167 -1
  393. package/src/evaluation/evaluation.mcp-tools.ts +155 -0
  394. package/src/evaluation/evaluation.module.ts +2 -1
  395. package/src/exam/dto/create-exam-question.dto.ts +8 -0
  396. package/src/exam/dto/create-exam.dto.ts +8 -0
  397. package/src/exam/dto/create-question-subject.dto.ts +12 -0
  398. package/src/exam/exam-attempt.service.ts +46 -14
  399. package/src/exam/exam.controller.ts +19 -0
  400. package/src/exam/exam.mcp-tools.ts +337 -0
  401. package/src/exam/exam.module.ts +2 -1
  402. package/src/exam/exam.service.ts +121 -0
  403. package/src/index.ts +9 -0
  404. package/src/instructor/instructor.mcp-tools.ts +243 -0
  405. package/src/instructor/instructor.module.ts +2 -1
  406. package/src/lms.module.ts +15 -1
  407. package/src/realtime/lms-realtime.controller.ts +12 -0
  408. package/src/realtime/lms-realtime.module.ts +12 -0
  409. package/src/realtime/lms-realtime.service.ts +98 -0
  410. package/src/realtime/lms-realtime.subscriber.ts +61 -0
  411. package/src/reports/reports.mcp-tools.ts +27 -0
  412. package/src/reports/reports.module.ts +2 -1
  413. package/src/training/training.mcp-tools.ts +128 -0
  414. package/src/training/training.module.ts +2 -1
@@ -0,0 +1,47 @@
1
+ import {
2
+ IsBoolean,
3
+ IsEnum,
4
+ IsInt,
5
+ IsOptional,
6
+ IsString,
7
+ MaxLength,
8
+ Min,
9
+ } from 'class-validator';
10
+ import {
11
+ ACHIEVEMENT_TYPE_VALUES,
12
+ type AchievementType,
13
+ } from './create-achievement.dto';
14
+
15
+ export class UpdateAchievementDto {
16
+ @IsOptional()
17
+ @IsString()
18
+ @MaxLength(255)
19
+ name?: string;
20
+
21
+ @IsOptional()
22
+ @IsString()
23
+ description?: string | null;
24
+
25
+ @IsOptional()
26
+ @IsInt()
27
+ @Min(1)
28
+ iconFileId?: number | null;
29
+
30
+ @IsOptional()
31
+ @IsInt()
32
+ @Min(1)
33
+ mainImageFileId?: number | null;
34
+
35
+ @IsOptional()
36
+ @IsEnum(ACHIEVEMENT_TYPE_VALUES)
37
+ type?: AchievementType;
38
+
39
+ @IsOptional()
40
+ @IsInt()
41
+ @Min(0)
42
+ bitCodesValue?: number;
43
+
44
+ @IsOptional()
45
+ @IsBoolean()
46
+ canReceiveMultipleTimes?: boolean;
47
+ }
@@ -0,0 +1,69 @@
1
+ import { Role } 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
+ Query,
14
+ } from '@nestjs/common';
15
+ import { BitcodeWalletService } from './bitcode-wallet.service';
16
+ import { CreateBitcodeWalletTransactionDto } from './dto/create-bitcode-wallet-transaction.dto';
17
+ import { CreateBitcodeWalletDto } from './dto/create-bitcode-wallet.dto';
18
+ import { UpdateBitcodeWalletDto } from './dto/update-bitcode-wallet.dto';
19
+
20
+ @Role()
21
+ @Controller('lms/bitcode-wallets')
22
+ export class BitcodeWalletController {
23
+ constructor(private readonly bitcodeWalletService: BitcodeWalletService) {}
24
+
25
+ @Get()
26
+ list(
27
+ @Query('page') page?: string,
28
+ @Query('pageSize') pageSize?: string,
29
+ @Query('search') search?: string
30
+ ) {
31
+ return this.bitcodeWalletService.list({
32
+ page: page ? Number(page) : 1,
33
+ pageSize: pageSize ? Number(pageSize) : 12,
34
+ search,
35
+ });
36
+ }
37
+
38
+ @Get(':id')
39
+ getById(@Param('id', ParseIntPipe) id: number) {
40
+ return this.bitcodeWalletService.getById(id);
41
+ }
42
+
43
+ @Post()
44
+ create(@Body() dto: CreateBitcodeWalletDto) {
45
+ return this.bitcodeWalletService.create(dto);
46
+ }
47
+
48
+ @Patch(':id')
49
+ update(
50
+ @Param('id', ParseIntPipe) id: number,
51
+ @Body() dto: UpdateBitcodeWalletDto
52
+ ) {
53
+ return this.bitcodeWalletService.update(id, dto);
54
+ }
55
+
56
+ @Delete(':id')
57
+ @HttpCode(HttpStatus.NO_CONTENT)
58
+ remove(@Param('id', ParseIntPipe) id: number) {
59
+ return this.bitcodeWalletService.delete(id);
60
+ }
61
+
62
+ @Post(':id/transactions')
63
+ createTransaction(
64
+ @Param('id', ParseIntPipe) id: number,
65
+ @Body() dto: CreateBitcodeWalletTransactionDto
66
+ ) {
67
+ return this.bitcodeWalletService.createTransaction(id, dto);
68
+ }
69
+ }
@@ -0,0 +1,107 @@
1
+ import { McpContext, McpTool } from '@hed-hog/core';
2
+ import { Injectable } from '@nestjs/common';
3
+ import { BitcodeWalletService } from './bitcode-wallet.service';
4
+
5
+ @Injectable()
6
+ export class LmsBitcodeWalletMcpTools {
7
+ constructor(private readonly bitcodeWalletService: BitcodeWalletService) {}
8
+
9
+ @McpTool({
10
+ name: 'lms.bitcode-wallet.list',
11
+ description: 'Lists LMS BitCode wallets with optional pagination and search by person name.',
12
+ inputSchema: {
13
+ type: 'object',
14
+ properties: {
15
+ page: { type: 'number', description: 'Page number (default: 1)' },
16
+ pageSize: { type: 'number', description: 'Items per page (default: 12)' },
17
+ search: { type: 'string', description: 'Search term (person name)' },
18
+ },
19
+ },
20
+ readOnly: true,
21
+ })
22
+ async listWallets(args: Record<string, any>, _context: McpContext): Promise<any> {
23
+ return this.bitcodeWalletService.list(args);
24
+ }
25
+
26
+ @McpTool({
27
+ name: 'lms.bitcode-wallet.get',
28
+ description: 'Returns a single BitCode wallet by ID including its transaction history.',
29
+ inputSchema: {
30
+ type: 'object',
31
+ properties: {
32
+ id: { type: 'number', description: 'Wallet ID' },
33
+ },
34
+ required: ['id'],
35
+ },
36
+ readOnly: true,
37
+ })
38
+ async getWallet(args: { id: number }, _context: McpContext): Promise<any> {
39
+ return this.bitcodeWalletService.getById(args.id);
40
+ }
41
+
42
+ @McpTool({
43
+ name: 'lms.bitcode-wallet.create',
44
+ description: 'Creates a new BitCode wallet for a person.',
45
+ inputSchema: {
46
+ type: 'object',
47
+ properties: {
48
+ personId: { type: 'number', description: 'Person ID to create the wallet for (required)' },
49
+ },
50
+ required: ['personId'],
51
+ },
52
+ })
53
+ async createWallet(args: { personId: number }, _context: McpContext): Promise<any> {
54
+ return this.bitcodeWalletService.create(args as any);
55
+ }
56
+
57
+ @McpTool({
58
+ name: 'lms.bitcode-wallet.update',
59
+ description: 'Updates the person linked to a BitCode wallet.',
60
+ inputSchema: {
61
+ type: 'object',
62
+ properties: {
63
+ id: { type: 'number', description: 'Wallet ID' },
64
+ personId: { type: 'number', description: 'New person ID' },
65
+ },
66
+ required: ['id'],
67
+ },
68
+ })
69
+ async updateWallet(args: { id: number; [key: string]: any }, _context: McpContext): Promise<any> {
70
+ const { id, ...dto } = args;
71
+ return this.bitcodeWalletService.update(id, dto as any);
72
+ }
73
+
74
+ @McpTool({
75
+ name: 'lms.bitcode-wallet.delete',
76
+ description: 'Deletes a BitCode wallet by ID.',
77
+ inputSchema: {
78
+ type: 'object',
79
+ properties: {
80
+ id: { type: 'number', description: 'Wallet ID' },
81
+ },
82
+ required: ['id'],
83
+ },
84
+ })
85
+ async deleteWallet(args: { id: number }, _context: McpContext): Promise<any> {
86
+ return this.bitcodeWalletService.delete(args.id);
87
+ }
88
+
89
+ @McpTool({
90
+ name: 'lms.bitcode-wallet.create-transaction',
91
+ description: 'Creates a credit or debit transaction on a BitCode wallet.',
92
+ inputSchema: {
93
+ type: 'object',
94
+ properties: {
95
+ id: { type: 'number', description: 'Wallet ID' },
96
+ type: { type: 'string', description: 'Transaction type', enum: ['credit', 'debit'] },
97
+ amount: { type: 'number', description: 'Transaction amount' },
98
+ description: { type: 'string', description: 'Transaction description' },
99
+ },
100
+ required: ['id', 'type', 'amount'],
101
+ },
102
+ })
103
+ async createTransaction(args: { id: number; [key: string]: any }, _context: McpContext): Promise<any> {
104
+ const { id, ...dto } = args;
105
+ return this.bitcodeWalletService.createTransaction(id, dto as any);
106
+ }
107
+ }
@@ -0,0 +1,13 @@
1
+ import { PrismaModule } from '@hed-hog/api-prisma';
2
+ import { forwardRef, Module } from '@nestjs/common';
3
+ import { BitcodeWalletController } from './bitcode-wallet.controller';
4
+ import { LmsBitcodeWalletMcpTools } from './bitcode-wallet.mcp-tools';
5
+ import { BitcodeWalletService } from './bitcode-wallet.service';
6
+
7
+ @Module({
8
+ imports: [forwardRef(() => PrismaModule)],
9
+ controllers: [BitcodeWalletController],
10
+ providers: [BitcodeWalletService, LmsBitcodeWalletMcpTools],
11
+ exports: [BitcodeWalletService],
12
+ })
13
+ export class BitcodeWalletModule {}
@@ -0,0 +1,361 @@
1
+ import { PrismaService } from '@hed-hog/api-prisma';
2
+ import {
3
+ ConflictException,
4
+ Injectable,
5
+ NotFoundException,
6
+ } from '@nestjs/common';
7
+ import { CreateBitcodeWalletTransactionDto } from './dto/create-bitcode-wallet-transaction.dto';
8
+ import { CreateBitcodeWalletDto } from './dto/create-bitcode-wallet.dto';
9
+ import { UpdateBitcodeWalletDto } from './dto/update-bitcode-wallet.dto';
10
+
11
+ type BitcodeWalletListRow = {
12
+ id: number;
13
+ personId: number;
14
+ personName: string | null;
15
+ currentBalance: number;
16
+ transactionCount: number;
17
+ totalCredits: number;
18
+ totalDebits: number;
19
+ createdAt: string | null;
20
+ updatedAt: string | null;
21
+ };
22
+
23
+ type BitcodeWalletSummaryRow = {
24
+ walletCount: number;
25
+ totalBalance: number;
26
+ transactionCount: number;
27
+ totalCredits: number;
28
+ totalDebits: number;
29
+ };
30
+
31
+ type BitcodeWalletTransactionRow = {
32
+ id: number;
33
+ walletId: number;
34
+ type: string;
35
+ amount: number;
36
+ description: string | null;
37
+ createdAt: string | null;
38
+ updatedAt: string | null;
39
+ };
40
+
41
+ @Injectable()
42
+ export class BitcodeWalletService {
43
+ constructor(private readonly prisma: PrismaService) {}
44
+
45
+ async list(params: { page?: number; pageSize?: number; search?: string }) {
46
+ const page = Math.max(Number(params.page) || 1, 1);
47
+ const pageSize = Math.max(Number(params.pageSize) || 12, 1);
48
+ const offset = (page - 1) * pageSize;
49
+ const search = params.search?.trim();
50
+
51
+ const values: unknown[] = [];
52
+ const where = ['1 = 1'];
53
+
54
+ if (search) {
55
+ const placeholder = this.param(values, `%${search}%`);
56
+ where.push(`p.name ILIKE ${placeholder}`);
57
+ }
58
+
59
+ const whereClause = where.join(' AND ');
60
+
61
+ const totalRow = await this.querySingle<{ total: string }>(
62
+ `SELECT COUNT(*)::text AS total
63
+ FROM bitcode_wallet bw
64
+ JOIN person p ON p.id = bw.person_id
65
+ WHERE ${whereClause}`,
66
+ values
67
+ );
68
+
69
+ const summaryRow =
70
+ (await this.querySingle<BitcodeWalletSummaryRow>(
71
+ `SELECT
72
+ COUNT(*)::int AS "walletCount",
73
+ COALESCE(SUM(bw.current_balance), 0)::int AS "totalBalance",
74
+ COALESCE(SUM(stats.transaction_count), 0)::int AS "transactionCount",
75
+ COALESCE(SUM(stats.total_credits), 0)::int AS "totalCredits",
76
+ COALESCE(SUM(stats.total_debits), 0)::int AS "totalDebits"
77
+ FROM bitcode_wallet bw
78
+ JOIN person p ON p.id = bw.person_id
79
+ LEFT JOIN LATERAL (
80
+ SELECT
81
+ COUNT(*)::int AS transaction_count,
82
+ COALESCE(SUM(CASE WHEN type = 'credit' THEN amount ELSE 0 END), 0)::int AS total_credits,
83
+ COALESCE(SUM(CASE WHEN type = 'debit' THEN amount ELSE 0 END), 0)::int AS total_debits
84
+ FROM bitcode_wallet_transaction bt
85
+ WHERE bt.wallet_id = bw.id
86
+ ) stats ON TRUE
87
+ WHERE ${whereClause}`,
88
+ values
89
+ )) ?? {
90
+ walletCount: 0,
91
+ totalBalance: 0,
92
+ transactionCount: 0,
93
+ totalCredits: 0,
94
+ totalDebits: 0,
95
+ };
96
+
97
+ const queryValues = [...values];
98
+ const limitPlaceholder = this.param(queryValues, pageSize);
99
+ const offsetPlaceholder = this.param(queryValues, offset);
100
+
101
+ const rows = await this.queryRows<BitcodeWalletListRow>(
102
+ `SELECT
103
+ bw.id,
104
+ bw.person_id AS "personId",
105
+ p.name AS "personName",
106
+ bw.current_balance AS "currentBalance",
107
+ COALESCE(stats.transaction_count, 0)::int AS "transactionCount",
108
+ COALESCE(stats.total_credits, 0)::int AS "totalCredits",
109
+ COALESCE(stats.total_debits, 0)::int AS "totalDebits",
110
+ bw.created_at::text AS "createdAt",
111
+ bw.updated_at::text AS "updatedAt"
112
+ FROM bitcode_wallet bw
113
+ JOIN person p ON p.id = bw.person_id
114
+ LEFT JOIN LATERAL (
115
+ SELECT
116
+ COUNT(*)::int AS transaction_count,
117
+ COALESCE(SUM(CASE WHEN type = 'credit' THEN amount ELSE 0 END), 0)::int AS total_credits,
118
+ COALESCE(SUM(CASE WHEN type = 'debit' THEN amount ELSE 0 END), 0)::int AS total_debits
119
+ FROM bitcode_wallet_transaction bt
120
+ WHERE bt.wallet_id = bw.id
121
+ ) stats ON TRUE
122
+ WHERE ${whereClause}
123
+ ORDER BY p.name ASC, bw.id ASC
124
+ LIMIT ${limitPlaceholder}
125
+ OFFSET ${offsetPlaceholder}`,
126
+ queryValues
127
+ );
128
+
129
+ const total = Number(totalRow?.total ?? 0);
130
+
131
+ return {
132
+ data: rows.map((row) => this.mapWalletRow(row)),
133
+ total,
134
+ page,
135
+ pageSize,
136
+ lastPage: Math.max(1, Math.ceil(total / pageSize)),
137
+ summary: {
138
+ walletCount: Number(summaryRow.walletCount ?? 0),
139
+ totalBalance: Number(summaryRow.totalBalance ?? 0),
140
+ transactionCount: Number(summaryRow.transactionCount ?? 0),
141
+ totalCredits: Number(summaryRow.totalCredits ?? 0),
142
+ totalDebits: Number(summaryRow.totalDebits ?? 0),
143
+ },
144
+ };
145
+ }
146
+
147
+ async getById(id: number) {
148
+ const row = await this.querySingle<BitcodeWalletListRow>(
149
+ `SELECT
150
+ bw.id,
151
+ bw.person_id AS "personId",
152
+ p.name AS "personName",
153
+ bw.current_balance AS "currentBalance",
154
+ COALESCE(stats.transaction_count, 0)::int AS "transactionCount",
155
+ COALESCE(stats.total_credits, 0)::int AS "totalCredits",
156
+ COALESCE(stats.total_debits, 0)::int AS "totalDebits",
157
+ bw.created_at::text AS "createdAt",
158
+ bw.updated_at::text AS "updatedAt"
159
+ FROM bitcode_wallet bw
160
+ JOIN person p ON p.id = bw.person_id
161
+ LEFT JOIN LATERAL (
162
+ SELECT
163
+ COUNT(*)::int AS transaction_count,
164
+ COALESCE(SUM(CASE WHEN type = 'credit' THEN amount ELSE 0 END), 0)::int AS total_credits,
165
+ COALESCE(SUM(CASE WHEN type = 'debit' THEN amount ELSE 0 END), 0)::int AS total_debits
166
+ FROM bitcode_wallet_transaction bt
167
+ WHERE bt.wallet_id = bw.id
168
+ ) stats ON TRUE
169
+ WHERE bw.id = $1`,
170
+ [id]
171
+ );
172
+
173
+ if (!row) {
174
+ throw new NotFoundException('BitCode wallet not found');
175
+ }
176
+
177
+ const transactions = await this.queryRows<BitcodeWalletTransactionRow>(
178
+ `SELECT
179
+ bt.id,
180
+ bt.wallet_id AS "walletId",
181
+ bt.type::text AS type,
182
+ bt.amount,
183
+ bt.description,
184
+ bt.created_at::text AS "createdAt",
185
+ bt.updated_at::text AS "updatedAt"
186
+ FROM bitcode_wallet_transaction bt
187
+ WHERE bt.wallet_id = $1
188
+ ORDER BY bt.created_at DESC, bt.id DESC`,
189
+ [id]
190
+ );
191
+
192
+ return {
193
+ ...this.mapWalletRow(row),
194
+ transactions: transactions.map((transaction) =>
195
+ this.mapTransactionRow(transaction)
196
+ ),
197
+ };
198
+ }
199
+
200
+ async create(dto: CreateBitcodeWalletDto) {
201
+ await this.ensurePersonExists(dto.personId);
202
+ await this.ensurePersonHasNoWallet(dto.personId);
203
+
204
+ const created = await this.querySingle<{ id: number }>(
205
+ `INSERT INTO bitcode_wallet (
206
+ person_id,
207
+ current_balance,
208
+ created_at,
209
+ updated_at
210
+ ) VALUES (
211
+ $1,
212
+ 0,
213
+ NOW(),
214
+ NOW()
215
+ )
216
+ RETURNING id`,
217
+ [dto.personId]
218
+ );
219
+
220
+ return this.getById(created.id);
221
+ }
222
+
223
+ async update(id: number, dto: UpdateBitcodeWalletDto) {
224
+ await this.ensureWalletExists(id);
225
+
226
+ if (dto.personId !== undefined) {
227
+ await this.ensurePersonExists(dto.personId);
228
+ await this.ensurePersonHasNoWallet(dto.personId, id);
229
+
230
+ await this.prisma.$executeRawUnsafe(
231
+ `UPDATE bitcode_wallet
232
+ SET person_id = $1,
233
+ updated_at = NOW()
234
+ WHERE id = $2`,
235
+ dto.personId,
236
+ id
237
+ );
238
+ }
239
+
240
+ return this.getById(id);
241
+ }
242
+
243
+ async delete(id: number) {
244
+ await this.ensureWalletExists(id);
245
+ await this.prisma.$executeRawUnsafe(`DELETE FROM bitcode_wallet WHERE id = $1`, id);
246
+ }
247
+
248
+ async createTransaction(
249
+ walletId: number,
250
+ dto: CreateBitcodeWalletTransactionDto
251
+ ) {
252
+ await this.ensureWalletExists(walletId);
253
+
254
+ await this.prisma.$executeRawUnsafe(
255
+ `INSERT INTO bitcode_wallet_transaction (
256
+ wallet_id,
257
+ type,
258
+ amount,
259
+ description,
260
+ created_at,
261
+ updated_at
262
+ ) VALUES (
263
+ $1,
264
+ $2::bitcode_wallet_transaction_type_7b37a2f43e_enum,
265
+ $3,
266
+ $4,
267
+ NOW(),
268
+ NOW()
269
+ )`,
270
+ walletId,
271
+ dto.type,
272
+ Number(dto.amount),
273
+ this.normalizeOptionalText(dto.description)
274
+ );
275
+
276
+ return this.getById(walletId);
277
+ }
278
+
279
+ private async ensurePersonExists(personId: number) {
280
+ const row = await this.querySingle<{ id: number }>(
281
+ `SELECT id FROM person WHERE id = $1`,
282
+ [personId]
283
+ );
284
+
285
+ if (!row) {
286
+ throw new NotFoundException('Person not found');
287
+ }
288
+ }
289
+
290
+ private async ensureWalletExists(id: number) {
291
+ const row = await this.querySingle<{ id: number }>(
292
+ `SELECT id FROM bitcode_wallet WHERE id = $1`,
293
+ [id]
294
+ );
295
+
296
+ if (!row) {
297
+ throw new NotFoundException('BitCode wallet not found');
298
+ }
299
+ }
300
+
301
+ private async ensurePersonHasNoWallet(personId: number, walletIdToIgnore?: number) {
302
+ const values: unknown[] = [personId];
303
+ let query = `SELECT id FROM bitcode_wallet WHERE person_id = $1`;
304
+
305
+ if (walletIdToIgnore) {
306
+ values.push(walletIdToIgnore);
307
+ query += ` AND id <> $2`;
308
+ }
309
+
310
+ const existing = await this.querySingle<{ id: number }>(query, values);
311
+
312
+ if (existing) {
313
+ throw new ConflictException('This person already has a BitCode wallet');
314
+ }
315
+ }
316
+
317
+ private mapWalletRow(row: BitcodeWalletListRow) {
318
+ return {
319
+ id: row.id,
320
+ personId: row.personId,
321
+ personName: row.personName ?? `Pessoa #${row.personId}`,
322
+ currentBalance: Number(row.currentBalance ?? 0),
323
+ transactionCount: Number(row.transactionCount ?? 0),
324
+ totalCredits: Number(row.totalCredits ?? 0),
325
+ totalDebits: Number(row.totalDebits ?? 0),
326
+ createdAt: row.createdAt,
327
+ updatedAt: row.updatedAt,
328
+ };
329
+ }
330
+
331
+ private mapTransactionRow(row: BitcodeWalletTransactionRow) {
332
+ return {
333
+ id: row.id,
334
+ walletId: row.walletId,
335
+ type: row.type,
336
+ amount: Number(row.amount ?? 0),
337
+ description: row.description,
338
+ createdAt: row.createdAt,
339
+ updatedAt: row.updatedAt,
340
+ };
341
+ }
342
+
343
+ private normalizeOptionalText(value?: string | null) {
344
+ const normalized = value?.trim();
345
+ return normalized ? normalized : null;
346
+ }
347
+
348
+ private param(values: unknown[], value: unknown) {
349
+ values.push(value);
350
+ return `$${values.length}`;
351
+ }
352
+
353
+ private async queryRows<T>(query: string, values: unknown[] = []) {
354
+ return this.prisma.$queryRawUnsafe<T[]>(query, ...values);
355
+ }
356
+
357
+ private async querySingle<T>(query: string, values: unknown[] = []) {
358
+ const rows = await this.prisma.$queryRawUnsafe<T[]>(query, ...values);
359
+ return rows[0] ?? null;
360
+ }
361
+ }
@@ -0,0 +1,27 @@
1
+ import {
2
+ IsEnum,
3
+ IsInt,
4
+ IsOptional,
5
+ IsString,
6
+ MaxLength,
7
+ Min,
8
+ } from 'class-validator';
9
+
10
+ export const BITCODE_TRANSACTION_TYPE_VALUES = ['credit', 'debit'] as const;
11
+
12
+ export type BitcodeTransactionType =
13
+ (typeof BITCODE_TRANSACTION_TYPE_VALUES)[number];
14
+
15
+ export class CreateBitcodeWalletTransactionDto {
16
+ @IsEnum(BITCODE_TRANSACTION_TYPE_VALUES)
17
+ type: BitcodeTransactionType;
18
+
19
+ @IsInt()
20
+ @Min(1)
21
+ amount: number;
22
+
23
+ @IsOptional()
24
+ @IsString()
25
+ @MaxLength(2000)
26
+ description?: string | null;
27
+ }
@@ -0,0 +1,7 @@
1
+ import { IsInt, Min } from 'class-validator';
2
+
3
+ export class CreateBitcodeWalletDto {
4
+ @IsInt()
5
+ @Min(1)
6
+ personId: number;
7
+ }
@@ -0,0 +1,28 @@
1
+ import {
2
+ IsEnum,
3
+ IsInt,
4
+ IsOptional,
5
+ IsString,
6
+ MaxLength,
7
+ Min,
8
+ } from 'class-validator';
9
+ import {
10
+ BITCODE_TRANSACTION_TYPE_VALUES,
11
+ type BitcodeTransactionType,
12
+ } from './create-bitcode-wallet-transaction.dto';
13
+
14
+ export class UpdateBitcodeWalletTransactionDto {
15
+ @IsOptional()
16
+ @IsEnum(BITCODE_TRANSACTION_TYPE_VALUES)
17
+ type?: BitcodeTransactionType;
18
+
19
+ @IsOptional()
20
+ @IsInt()
21
+ @Min(1)
22
+ amount?: number;
23
+
24
+ @IsOptional()
25
+ @IsString()
26
+ @MaxLength(2000)
27
+ description?: string | null;
28
+ }
@@ -0,0 +1,8 @@
1
+ import { IsInt, IsOptional, Min } from 'class-validator';
2
+
3
+ export class UpdateBitcodeWalletDto {
4
+ @IsOptional()
5
+ @IsInt()
6
+ @Min(1)
7
+ personId?: number;
8
+ }