@hed-hog/lms 0.0.364 → 0.0.366

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 (290) hide show
  1. package/dist/bitcode-wallet/bitcode-wallet.service.d.ts +1 -0
  2. package/dist/bitcode-wallet/bitcode-wallet.service.d.ts.map +1 -1
  3. package/dist/bitcode-wallet/bitcode-wallet.service.js +22 -3
  4. package/dist/bitcode-wallet/bitcode-wallet.service.js.map +1 -1
  5. package/dist/class-group/class-group.controller.d.ts +1 -0
  6. package/dist/class-group/class-group.controller.d.ts.map +1 -1
  7. package/dist/class-group/class-group.service.d.ts +1 -0
  8. package/dist/class-group/class-group.service.d.ts.map +1 -1
  9. package/dist/course/course-export-scorm12-worker.service.d.ts +21 -0
  10. package/dist/course/course-export-scorm12-worker.service.d.ts.map +1 -0
  11. package/dist/course/course-export-scorm12-worker.service.js +109 -0
  12. package/dist/course/course-export-scorm12-worker.service.js.map +1 -0
  13. package/dist/course/course-export-scorm12.service.d.ts +42 -0
  14. package/dist/course/course-export-scorm12.service.d.ts.map +1 -0
  15. package/dist/course/course-export-scorm12.service.js +628 -0
  16. package/dist/course/course-export-scorm12.service.js.map +1 -0
  17. package/dist/course/course-export.service.d.ts +84 -0
  18. package/dist/course/course-export.service.d.ts.map +1 -0
  19. package/dist/course/course-export.service.js +237 -0
  20. package/dist/course/course-export.service.js.map +1 -0
  21. package/dist/course/course-structure.controller.d.ts +20 -10
  22. package/dist/course/course-structure.controller.d.ts.map +1 -1
  23. package/dist/course/course-structure.controller.js +20 -4
  24. package/dist/course/course-structure.controller.js.map +1 -1
  25. package/dist/course/course-structure.service.d.ts +12 -4
  26. package/dist/course/course-structure.service.d.ts.map +1 -1
  27. package/dist/course/course-structure.service.js +98 -23
  28. package/dist/course/course-structure.service.js.map +1 -1
  29. package/dist/course/course-video-agent-pipeline.service.d.ts +70 -0
  30. package/dist/course/course-video-agent-pipeline.service.d.ts.map +1 -0
  31. package/dist/course/course-video-agent-pipeline.service.js +398 -0
  32. package/dist/course/course-video-agent-pipeline.service.js.map +1 -0
  33. package/dist/course/course-video-hls.service.d.ts +71 -0
  34. package/dist/course/course-video-hls.service.d.ts.map +1 -0
  35. package/dist/course/course-video-hls.service.js +784 -0
  36. package/dist/course/course-video-hls.service.js.map +1 -0
  37. package/dist/course/course.controller.d.ts +47 -13
  38. package/dist/course/course.controller.d.ts.map +1 -1
  39. package/dist/course/course.controller.js +40 -26
  40. package/dist/course/course.controller.js.map +1 -1
  41. package/dist/course/course.mcp-tools.js +1 -1
  42. package/dist/course/course.mcp-tools.js.map +1 -1
  43. package/dist/course/course.module.d.ts.map +1 -1
  44. package/dist/course/course.module.js +16 -0
  45. package/dist/course/course.module.js.map +1 -1
  46. package/dist/course/course.service.d.ts +8 -9
  47. package/dist/course/course.service.d.ts.map +1 -1
  48. package/dist/course/course.service.js +93 -50
  49. package/dist/course/course.service.js.map +1 -1
  50. package/dist/course/dto/cleanup-course-storage.dto.d.ts +1 -1
  51. package/dist/course/dto/cleanup-course-storage.dto.d.ts.map +1 -1
  52. package/dist/course/dto/cleanup-course-storage.dto.js +1 -0
  53. package/dist/course/dto/cleanup-course-storage.dto.js.map +1 -1
  54. package/dist/course/dto/cleanup-upload-history.dto.d.ts +1 -1
  55. package/dist/course/dto/cleanup-upload-history.dto.d.ts.map +1 -1
  56. package/dist/course/dto/cleanup-upload-history.dto.js +1 -1
  57. package/dist/course/dto/cleanup-upload-history.dto.js.map +1 -1
  58. package/dist/course/dto/create-course-bulk-job.dto.d.ts +2 -1
  59. package/dist/course/dto/create-course-bulk-job.dto.d.ts.map +1 -1
  60. package/dist/course/dto/create-course-bulk-job.dto.js +6 -1
  61. package/dist/course/dto/create-course-bulk-job.dto.js.map +1 -1
  62. package/dist/course/dto/create-course-export.dto.d.ts +14 -0
  63. package/dist/course/dto/create-course-export.dto.d.ts.map +1 -0
  64. package/dist/course/dto/create-course-export.dto.js +71 -0
  65. package/dist/course/dto/create-course-export.dto.js.map +1 -0
  66. package/dist/course/dto/create-course-structure-lesson.dto.d.ts +2 -2
  67. package/dist/course/dto/create-course-structure-lesson.dto.d.ts.map +1 -1
  68. package/dist/course/dto/create-course-structure-lesson.dto.js +3 -2
  69. package/dist/course/dto/create-course-structure-lesson.dto.js.map +1 -1
  70. package/dist/course/ffmpeg.util.d.ts +10 -0
  71. package/dist/course/ffmpeg.util.d.ts.map +1 -0
  72. package/dist/course/ffmpeg.util.js +79 -0
  73. package/dist/course/ffmpeg.util.js.map +1 -0
  74. package/dist/course/lms-bulk-upload-automation.service.d.ts +18 -1
  75. package/dist/course/lms-bulk-upload-automation.service.d.ts.map +1 -1
  76. package/dist/course/lms-bulk-upload-automation.service.js +106 -8
  77. package/dist/course/lms-bulk-upload-automation.service.js.map +1 -1
  78. package/dist/course/lms-bulk-upload-infra.service.d.ts +1 -0
  79. package/dist/course/lms-bulk-upload-infra.service.d.ts.map +1 -1
  80. package/dist/course/lms-bulk-upload-infra.service.js +32 -8
  81. package/dist/course/lms-bulk-upload-infra.service.js.map +1 -1
  82. package/dist/course/lms-bulk-upload.controller.d.ts +30 -3
  83. package/dist/course/lms-bulk-upload.controller.d.ts.map +1 -1
  84. package/dist/course/lms-bulk-upload.controller.js +43 -2
  85. package/dist/course/lms-bulk-upload.controller.js.map +1 -1
  86. package/dist/course/lms-bulk-upload.service.d.ts +11 -0
  87. package/dist/course/lms-bulk-upload.service.d.ts.map +1 -1
  88. package/dist/course/lms-bulk-upload.service.js +59 -6
  89. package/dist/course/lms-bulk-upload.service.js.map +1 -1
  90. package/dist/course/lms-setting.controller.d.ts +2 -1
  91. package/dist/course/lms-setting.controller.d.ts.map +1 -1
  92. package/dist/course/lms-setting.controller.js +4 -2
  93. package/dist/course/lms-setting.controller.js.map +1 -1
  94. package/dist/course/scorm12-schemas.d.ts +4 -0
  95. package/dist/course/scorm12-schemas.d.ts.map +1 -0
  96. package/dist/course/scorm12-schemas.js +9 -0
  97. package/dist/course/scorm12-schemas.js.map +1 -0
  98. package/dist/enterprise/training/training-admin.controller.d.ts +2 -0
  99. package/dist/enterprise/training/training-admin.controller.d.ts.map +1 -1
  100. package/dist/enterprise/training/training-admin.service.d.ts +2 -0
  101. package/dist/enterprise/training/training-admin.service.d.ts.map +1 -1
  102. package/dist/enterprise/training/training-student.service.d.ts +51 -0
  103. package/dist/enterprise/training/training-student.service.d.ts.map +1 -1
  104. package/dist/enterprise/training/training-student.service.js +217 -4
  105. package/dist/enterprise/training/training-student.service.js.map +1 -1
  106. package/dist/evaluation/evaluation.service.d.ts +18 -0
  107. package/dist/evaluation/evaluation.service.d.ts.map +1 -1
  108. package/dist/evaluation/evaluation.service.js +125 -0
  109. package/dist/evaluation/evaluation.service.js.map +1 -1
  110. package/dist/exam/dto/create-standalone-question.dto.d.ts +12 -0
  111. package/dist/exam/dto/create-standalone-question.dto.d.ts.map +1 -0
  112. package/dist/exam/dto/create-standalone-question.dto.js +70 -0
  113. package/dist/exam/dto/create-standalone-question.dto.js.map +1 -0
  114. package/dist/exam/exam.module.d.ts.map +1 -1
  115. package/dist/exam/exam.module.js +2 -1
  116. package/dist/exam/exam.module.js.map +1 -1
  117. package/dist/exam/exam.service.d.ts +21 -0
  118. package/dist/exam/exam.service.d.ts.map +1 -1
  119. package/dist/exam/exam.service.js +80 -0
  120. package/dist/exam/exam.service.js.map +1 -1
  121. package/dist/exam/question.controller.d.ts +27 -0
  122. package/dist/exam/question.controller.d.ts.map +1 -0
  123. package/dist/exam/question.controller.js +53 -0
  124. package/dist/exam/question.controller.js.map +1 -0
  125. package/dist/lesson-xp-map/lesson-xp-ai-calculation.service.d.ts +4 -0
  126. package/dist/lesson-xp-map/lesson-xp-ai-calculation.service.d.ts.map +1 -1
  127. package/dist/lesson-xp-map/lesson-xp-ai-calculation.service.js +161 -25
  128. package/dist/lesson-xp-map/lesson-xp-ai-calculation.service.js.map +1 -1
  129. package/dist/libraries/lms/tsconfig.tsbuildinfo +1 -1
  130. package/dist/lms-commerce-access.subscriber.d.ts +11 -0
  131. package/dist/lms-commerce-access.subscriber.d.ts.map +1 -0
  132. package/dist/lms-commerce-access.subscriber.js +74 -0
  133. package/dist/lms-commerce-access.subscriber.js.map +1 -0
  134. package/dist/lms.module.d.ts.map +1 -1
  135. package/dist/lms.module.js +16 -5
  136. package/dist/lms.module.js.map +1 -1
  137. package/dist/platforma/dto/heartbeat.dto.d.ts +9 -0
  138. package/dist/platforma/dto/heartbeat.dto.d.ts.map +1 -0
  139. package/dist/platforma/dto/heartbeat.dto.js +50 -0
  140. package/dist/platforma/dto/heartbeat.dto.js.map +1 -0
  141. package/dist/platforma/handlers/emit-certificate.handler.d.ts +27 -0
  142. package/dist/platforma/handlers/emit-certificate.handler.d.ts.map +1 -0
  143. package/dist/platforma/handlers/emit-certificate.handler.js +117 -0
  144. package/dist/platforma/handlers/emit-certificate.handler.js.map +1 -0
  145. package/dist/platforma/handlers/lesson-heartbeat.handler.d.ts +31 -0
  146. package/dist/platforma/handlers/lesson-heartbeat.handler.d.ts.map +1 -0
  147. package/dist/platforma/handlers/lesson-heartbeat.handler.js +281 -0
  148. package/dist/platforma/handlers/lesson-heartbeat.handler.js.map +1 -0
  149. package/dist/platforma/platforma-heartbeat.service.d.ts +10 -0
  150. package/dist/platforma/platforma-heartbeat.service.d.ts.map +1 -0
  151. package/dist/platforma/platforma-heartbeat.service.js +50 -0
  152. package/dist/platforma/platforma-heartbeat.service.js.map +1 -0
  153. package/dist/platforma/platforma-performance.service.d.ts +121 -0
  154. package/dist/platforma/platforma-performance.service.d.ts.map +1 -0
  155. package/dist/platforma/platforma-performance.service.js +500 -0
  156. package/dist/platforma/platforma-performance.service.js.map +1 -0
  157. package/dist/platforma/platforma-search.service.d.ts +21 -0
  158. package/dist/platforma/platforma-search.service.d.ts.map +1 -0
  159. package/dist/platforma/platforma-search.service.js +64 -0
  160. package/dist/platforma/platforma-search.service.js.map +1 -0
  161. package/dist/platforma/platforma-video.service.d.ts +39 -0
  162. package/dist/platforma/platforma-video.service.d.ts.map +1 -0
  163. package/dist/platforma/platforma-video.service.js +301 -0
  164. package/dist/platforma/platforma-video.service.js.map +1 -0
  165. package/dist/platforma/platforma.controller.d.ts +209 -1
  166. package/dist/platforma/platforma.controller.d.ts.map +1 -1
  167. package/dist/platforma/platforma.controller.js +208 -2
  168. package/dist/platforma/platforma.controller.js.map +1 -1
  169. package/dist/realtime/lms-realtime.controller.d.ts +2 -0
  170. package/dist/realtime/lms-realtime.controller.d.ts.map +1 -1
  171. package/dist/realtime/lms-realtime.controller.js +31 -0
  172. package/dist/realtime/lms-realtime.controller.js.map +1 -1
  173. package/dist/realtime/lms-realtime.service.d.ts +1 -1
  174. package/dist/realtime/lms-realtime.service.d.ts.map +1 -1
  175. package/dist/realtime/lms-realtime.service.js.map +1 -1
  176. package/dist/student-xp/dto/grant-skill-card-xp.dto.d.ts +5 -0
  177. package/dist/student-xp/dto/grant-skill-card-xp.dto.d.ts.map +1 -0
  178. package/dist/student-xp/dto/grant-skill-card-xp.dto.js +26 -0
  179. package/dist/student-xp/dto/grant-skill-card-xp.dto.js.map +1 -0
  180. package/dist/student-xp/student-xp.controller.d.ts +15 -0
  181. package/dist/student-xp/student-xp.controller.d.ts.map +1 -1
  182. package/dist/student-xp/student-xp.controller.js +24 -0
  183. package/dist/student-xp/student-xp.controller.js.map +1 -1
  184. package/dist/student-xp/student-xp.service.d.ts +16 -0
  185. package/dist/student-xp/student-xp.service.d.ts.map +1 -1
  186. package/dist/student-xp/student-xp.service.js +51 -1
  187. package/dist/student-xp/student-xp.service.js.map +1 -1
  188. package/hedhog/data/evaluation_topic.yaml +17 -0
  189. package/hedhog/data/menu.yaml +0 -17
  190. package/hedhog/data/queue_definition.yaml +48 -0
  191. package/hedhog/data/route.yaml +94 -124
  192. package/hedhog/data/setting_group.yaml +19 -19
  193. package/hedhog/frontend/app/bulk-upload-sessions/page.tsx.ejs +337 -41
  194. package/hedhog/frontend/app/certificates/models/page.tsx.ejs +182 -29
  195. package/hedhog/frontend/app/classes/_components/classes-calendar-view.tsx.ejs +277 -0
  196. package/hedhog/frontend/app/classes/page.tsx.ejs +127 -20
  197. package/hedhog/frontend/app/courses/[id]/page.tsx.ejs +69 -4
  198. package/hedhog/frontend/app/courses/[id]/structure/_components/course-export-sheet.tsx.ejs +420 -0
  199. package/hedhog/frontend/app/courses/[id]/structure/_components/course-exports-tab.tsx.ejs +308 -0
  200. package/hedhog/frontend/app/courses/[id]/structure/_components/course-overview-tab.tsx.ejs +158 -45
  201. package/hedhog/frontend/app/courses/[id]/structure/_components/course-xp-overview-tab.tsx.ejs +13 -13
  202. package/hedhog/frontend/app/courses/[id]/structure/_components/detail-lesson-xp-tab.tsx.ejs +11 -23
  203. package/hedhog/frontend/app/courses/[id]/structure/_components/detail-lesson.tsx.ejs +51 -63
  204. package/hedhog/frontend/app/courses/[id]/structure/_components/detail-panel.tsx.ejs +8 -3
  205. package/hedhog/frontend/app/courses/[id]/structure/_components/detail-session.tsx.ejs +31 -8
  206. package/hedhog/frontend/app/courses/[id]/structure/_components/drag-overlay.tsx.ejs +16 -9
  207. package/hedhog/frontend/app/courses/[id]/structure/_components/editor-course.tsx.ejs +201 -401
  208. package/hedhog/frontend/app/courses/[id]/structure/_components/editor-lesson.tsx.ejs +378 -690
  209. package/hedhog/frontend/app/courses/[id]/structure/_components/mock-data.ts.ejs +1 -2
  210. package/hedhog/frontend/app/courses/[id]/structure/_components/tree-row-lesson.tsx.ejs +3 -9
  211. package/hedhog/frontend/app/courses/[id]/structure/_components/types.ts.ejs +1 -1
  212. package/hedhog/frontend/app/courses/[id]/structure/_components/xp-premium-pills.tsx.ejs +1 -8
  213. package/hedhog/frontend/app/courses/[id]/structure/_data/adapters/course-structure.adapter.ts.ejs +6 -10
  214. package/hedhog/frontend/app/courses/[id]/structure/_data/services/course-structure.service.ts.ejs +49 -0
  215. package/hedhog/frontend/app/courses/[id]/structure/_data/types/api-course.types.ts.ejs +4 -3
  216. package/hedhog/frontend/app/courses/[id]/structure/_data/use-course-content-overview.ts.ejs +2 -1
  217. package/hedhog/frontend/app/courses/[id]/structure/_data/use-course-exports.ts.ejs +106 -0
  218. package/hedhog/frontend/app/courses/[id]/structure/_data/use-course-structure-mutations.ts.ejs +28 -1
  219. package/hedhog/frontend/app/courses/[id]/structure/_data/use-lms-settings-query.ts.ejs +0 -2
  220. package/hedhog/frontend/app/courses/page.tsx.ejs +85 -9
  221. package/hedhog/frontend/app/enterprise/[id]/page.tsx.ejs +6 -0
  222. package/hedhog/frontend/app/enterprise/_components/enterprise-classes-calendar-tab.tsx.ejs +264 -0
  223. package/hedhog/frontend/app/enterprise/page.tsx.ejs +104 -47
  224. package/hedhog/frontend/app/exams/page.tsx.ejs +38 -4
  225. package/hedhog/frontend/app/instructors/page.tsx.ejs +87 -46
  226. package/hedhog/frontend/app/paths/page.tsx.ejs +38 -4
  227. package/hedhog/frontend/app/training/page.tsx.ejs +38 -4
  228. package/hedhog/frontend/messages/en.json +44 -28
  229. package/hedhog/frontend/messages/pt.json +47 -29
  230. package/hedhog/table/course_enrollment.yaml +3 -0
  231. package/hedhog/table/course_export.yaml +62 -0
  232. package/hedhog/table/lesson_view_event.yaml +66 -0
  233. package/package.json +14 -9
  234. package/src/bitcode-wallet/bitcode-wallet.service.ts +43 -4
  235. package/src/course/course-export-scorm12-worker.service.ts +124 -0
  236. package/src/course/course-export-scorm12.service.ts +668 -0
  237. package/src/course/course-export.service.ts +280 -0
  238. package/src/course/course-structure.controller.ts +16 -2
  239. package/src/course/course-structure.service.ts +100 -7
  240. package/src/course/course-video-agent-pipeline.service.ts +471 -0
  241. package/src/course/course-video-hls.service.ts +966 -0
  242. package/src/course/course.controller.ts +33 -19
  243. package/src/course/course.mcp-tools.ts +1 -1
  244. package/src/course/course.module.ts +16 -0
  245. package/src/course/course.service.ts +119 -61
  246. package/src/course/dto/cleanup-course-storage.dto.ts +1 -0
  247. package/src/course/dto/cleanup-upload-history.dto.ts +1 -1
  248. package/src/course/dto/create-course-bulk-job.dto.ts +7 -3
  249. package/src/course/dto/create-course-export.dto.ts +56 -0
  250. package/src/course/dto/create-course-structure-lesson.dto.ts +4 -3
  251. package/src/course/ffmpeg.util.ts +65 -0
  252. package/src/course/lms-bulk-upload-automation.service.ts +156 -6
  253. package/src/course/lms-bulk-upload-infra.service.ts +39 -6
  254. package/src/course/lms-bulk-upload.controller.ts +32 -2
  255. package/src/course/lms-bulk-upload.service.ts +70 -7
  256. package/src/course/lms-setting.controller.ts +4 -2
  257. package/src/course/scorm12-schemas.ts +9 -0
  258. package/src/enterprise/training/training-student.service.ts +221 -2
  259. package/src/evaluation/evaluation.service.ts +123 -0
  260. package/src/exam/dto/create-standalone-question.dto.ts +66 -0
  261. package/src/exam/exam.module.ts +2 -1
  262. package/src/exam/exam.service.ts +86 -0
  263. package/src/exam/question.controller.ts +28 -0
  264. package/src/lesson-xp-map/lesson-xp-ai-calculation.service.ts +205 -31
  265. package/src/lms-commerce-access.subscriber.ts +88 -0
  266. package/src/lms.module.ts +16 -5
  267. package/src/platforma/dto/heartbeat.dto.ts +30 -0
  268. package/src/platforma/handlers/emit-certificate.handler.ts +117 -0
  269. package/src/platforma/handlers/lesson-heartbeat.handler.ts +343 -0
  270. package/src/platforma/platforma-heartbeat.service.ts +33 -0
  271. package/src/platforma/platforma-performance.service.ts +606 -0
  272. package/src/platforma/platforma-search.service.ts +48 -0
  273. package/src/platforma/platforma-video.service.ts +346 -0
  274. package/src/platforma/platforma.controller.ts +137 -1
  275. package/src/platforma/platforma.service.ts +268 -268
  276. package/src/realtime/lms-realtime.controller.ts +27 -1
  277. package/src/realtime/lms-realtime.service.ts +2 -1
  278. package/src/student-xp/dto/grant-skill-card-xp.dto.ts +10 -0
  279. package/src/student-xp/student-xp.controller.ts +18 -2
  280. package/src/student-xp/student-xp.service.ts +84 -2
  281. package/hedhog/data/video_resolution_profile.yaml +0 -7
  282. package/hedhog/frontend/app/video-resolution-profiles/page.tsx.ejs +0 -607
  283. package/hedhog/table/course_video_resolution_profile.yaml +0 -22
  284. package/hedhog/table/video_resolution_profile.yaml +0 -18
  285. package/src/video-resolution-profile/dto/create-video-resolution-profile.dto.ts +0 -16
  286. package/src/video-resolution-profile/dto/update-video-resolution-profile.dto.ts +0 -16
  287. package/src/video-resolution-profile/video-resolution-profile.controller.ts +0 -62
  288. package/src/video-resolution-profile/video-resolution-profile.mcp-tools.ts +0 -128
  289. package/src/video-resolution-profile/video-resolution-profile.module.ts +0 -13
  290. package/src/video-resolution-profile/video-resolution-profile.service.ts +0 -117
@@ -0,0 +1,346 @@
1
+ import { FileService } from '@hed-hog/core';
2
+ import { PrismaService } from '@hed-hog/api-prisma';
3
+ import {
4
+ forwardRef,
5
+ Inject,
6
+ Injectable,
7
+ Logger,
8
+ NotFoundException,
9
+ UnauthorizedException,
10
+ } from '@nestjs/common';
11
+ import { JwtService } from '@nestjs/jwt';
12
+ import type { Request, Response } from 'express';
13
+
14
+ @Injectable()
15
+ export class PlatformaVideoService {
16
+ private readonly logger = new Logger(PlatformaVideoService.name);
17
+ constructor(
18
+ @Inject(forwardRef(() => PrismaService))
19
+ private readonly prisma: PrismaService,
20
+ @Inject(forwardRef(() => FileService))
21
+ private readonly fileService: FileService,
22
+ @Inject(forwardRef(() => JwtService))
23
+ private readonly jwtService: JwtService,
24
+ ) {}
25
+
26
+ async getVideoOptions(lessonId: number) {
27
+ const lesson = await this.prisma.course_lesson.findFirst({
28
+ where: { id: lessonId, published: true },
29
+ select: {
30
+ id: true,
31
+ course_lesson_file: {
32
+ where: { file_id: { not: null } },
33
+ select: { id: true, title: true, type: true, file: { select: { id: true } } },
34
+ },
35
+ },
36
+ });
37
+
38
+ if (!lesson) throw new NotFoundException('Lesson not found');
39
+
40
+ // HLS takes priority over legacy MP4 profiles
41
+ const hlsFile = lesson.course_lesson_file.find((f) => f.type === 'video_hls' && f.file?.id);
42
+ if (hlsFile) {
43
+ return [{ type: 'video_hls', label: 'HLS', fileId: hlsFile.file!.id! }];
44
+ }
45
+
46
+ const videoFiles = lesson.course_lesson_file.filter(
47
+ (f) => f.type === 'video_original' || f.type?.startsWith('video_profile:'),
48
+ );
49
+
50
+ if (videoFiles.length === 0) return [];
51
+
52
+ return videoFiles
53
+ .filter((f) => f.file?.id)
54
+ .map((f) => {
55
+ const profileMatch = f.type?.match(/^video_profile:(\d+)$/);
56
+ const profileId = profileMatch ? Number(profileMatch[1]) : null;
57
+ const label =
58
+ f.type === 'video_original'
59
+ ? 'Original'
60
+ : profileId !== null
61
+ ? f.title || `Perfil ${profileId}`
62
+ : f.title || f.type || 'Vídeo';
63
+
64
+ return { type: f.type!, label, fileId: f.file!.id! };
65
+ });
66
+ }
67
+
68
+ async generateVideoToken(lessonId: number, quality?: string) {
69
+ const options = await this.getVideoOptions(lessonId);
70
+ if (options.length === 0) throw new NotFoundException('No video available for this lesson');
71
+
72
+ const selected = quality
73
+ ? (options.find((o) => o.type === quality) ?? options[0])
74
+ : options[0];
75
+
76
+ const token = this.jwtService.sign(
77
+ { fileId: selected.fileId, lessonId },
78
+ { secret: process.env.JWT_SECRET, expiresIn: '4h' },
79
+ );
80
+
81
+ return { token, label: selected.label, type: selected.type, expiresIn: 14400 };
82
+ }
83
+
84
+ async streamVideo(token: string, req: Request, res: Response) {
85
+ let fileId: number;
86
+ try {
87
+ const payload = this.jwtService.verify<{ fileId: number }>(token, {
88
+ secret: process.env.JWT_SECRET,
89
+ });
90
+ fileId = payload.fileId;
91
+ } catch {
92
+ throw new UnauthorizedException('Invalid or expired video token');
93
+ }
94
+
95
+ const rangeHeader = req.headers['range'];
96
+ let start: number | undefined;
97
+ let end: number | undefined;
98
+
99
+ if (rangeHeader) {
100
+ const match = rangeHeader.match(/bytes=(\d+)-(\d*)/);
101
+ if (match) {
102
+ start = parseInt(match[1], 10);
103
+ end = match[2] ? parseInt(match[2], 10) : undefined;
104
+ }
105
+ }
106
+
107
+ const result = await this.fileService.openReadStreamWithRange(fileId, start, end);
108
+ if (!result) throw new NotFoundException('Video file not found');
109
+
110
+ const { stream, totalSize, start: resolvedStart, end: resolvedEnd, contentLength, mimetype } = result;
111
+
112
+ const headers: Record<string, string | number> = {
113
+ 'Content-Type': mimetype,
114
+ 'Content-Disposition': 'inline',
115
+ 'Accept-Ranges': 'bytes',
116
+ 'Cache-Control': 'no-store',
117
+ };
118
+
119
+ if (start !== undefined) {
120
+ headers['Content-Range'] = `bytes ${resolvedStart}-${resolvedEnd ?? totalSize - 1}/${totalSize || '*'}`;
121
+ headers['Content-Length'] = contentLength;
122
+ res.writeHead(206, headers);
123
+ } else {
124
+ if (totalSize > 0) headers['Content-Length'] = totalSize;
125
+ res.writeHead(200, headers);
126
+ }
127
+
128
+ stream.pipe(res);
129
+ }
130
+
131
+ // ---------------------------------------------------------------------------
132
+ // HLS endpoints
133
+ // ---------------------------------------------------------------------------
134
+
135
+ async generateHlsToken(lessonId: number) {
136
+ const lesson = await this.prisma.course_lesson.findFirst({
137
+ where: { id: lessonId, published: true },
138
+ select: {
139
+ id: true,
140
+ course_lesson_file: {
141
+ where: { type: 'video_hls', file_id: { not: null } },
142
+ select: { id: true },
143
+ },
144
+ },
145
+ });
146
+
147
+ if (!lesson) throw new NotFoundException('Lesson not found');
148
+ if (lesson.course_lesson_file.length === 0) throw new NotFoundException('No HLS stream available for this lesson');
149
+
150
+ const token = this.jwtService.sign(
151
+ { lessonId },
152
+ { secret: process.env.JWT_SECRET, expiresIn: '4h' },
153
+ );
154
+
155
+ return { token, expiresIn: 14400 };
156
+ }
157
+
158
+ private verifyHlsToken(token: string): { lessonId: number } {
159
+ try {
160
+ return this.jwtService.verify<{ lessonId: number }>(token, {
161
+ secret: process.env.JWT_SECRET,
162
+ });
163
+ } catch {
164
+ throw new UnauthorizedException('Invalid or expired HLS token');
165
+ }
166
+ }
167
+
168
+ async serveHlsMasterManifest(token: string, req: Request, res: Response) {
169
+ const { lessonId } = this.verifyHlsToken(token);
170
+
171
+ const manifestFile = await (this.prisma as any).file.findFirst({
172
+ where: {
173
+ location: `lms/lessons/hls/${lessonId}`,
174
+ filename: 'master.m3u8',
175
+ },
176
+ select: { id: true },
177
+ });
178
+
179
+ if (!manifestFile) throw new NotFoundException('HLS master manifest not found');
180
+
181
+ const { stream } = await this.fileService.openReadStreamById(manifestFile.id);
182
+
183
+ const chunks: Buffer[] = [];
184
+ await new Promise<void>((resolve, reject) => {
185
+ stream.on('data', (chunk: Buffer) => chunks.push(chunk));
186
+ stream.on('end', resolve);
187
+ stream.on('error', reject);
188
+ });
189
+
190
+ const baseUrl = `${req.protocol}://${req.get('host')}`;
191
+ const content = Buffer.concat(chunks).toString('utf8');
192
+
193
+ // Rewrite relative variant playlist paths to absolute authenticated URLs
194
+ const rewritten = content.replace(
195
+ /^(stream_\d+\/playlist\.m3u8)$/gm,
196
+ (_, variantPath) => `${baseUrl}/lms/platforma/hls/${token}/${variantPath}`,
197
+ );
198
+
199
+ res.set({
200
+ 'Content-Type': 'application/x-mpegURL',
201
+ 'Cache-Control': 'no-store',
202
+ 'Access-Control-Allow-Origin': '*',
203
+ });
204
+ res.send(rewritten);
205
+ }
206
+
207
+ async serveHlsVariantPlaylist(token: string, variant: string, req: Request, res: Response) {
208
+ const { lessonId } = this.verifyHlsToken(token);
209
+
210
+ const playlistFile = await (this.prisma as any).file.findFirst({
211
+ where: {
212
+ location: `lms/lessons/hls/${lessonId}/${variant}`,
213
+ filename: 'playlist.m3u8',
214
+ },
215
+ select: { id: true },
216
+ });
217
+
218
+ if (!playlistFile) throw new NotFoundException('HLS variant playlist not found');
219
+
220
+ const { stream } = await this.fileService.openReadStreamById(playlistFile.id);
221
+
222
+ const chunks: Buffer[] = [];
223
+ await new Promise<void>((resolve, reject) => {
224
+ stream.on('data', (chunk: Buffer) => chunks.push(chunk));
225
+ stream.on('end', resolve);
226
+ stream.on('error', reject);
227
+ });
228
+
229
+ const baseUrl = `${req.protocol}://${req.get('host')}`;
230
+ const content = Buffer.concat(chunks).toString('utf8');
231
+
232
+ // Rewrite relative segment paths to absolute authenticated URLs
233
+ const rewritten = content.replace(
234
+ /^(seg\d+\.ts)$/gm,
235
+ (_, segName) => `${baseUrl}/lms/platforma/hls/${token}/${variant}/${segName}`,
236
+ );
237
+
238
+ res.set({
239
+ 'Content-Type': 'application/x-mpegURL',
240
+ 'Cache-Control': 'no-store',
241
+ 'Access-Control-Allow-Origin': '*',
242
+ });
243
+ res.send(rewritten);
244
+ }
245
+
246
+ async serveHlsSegment(token: string, variant: string, segment: string, res: Response) {
247
+ const { lessonId } = this.verifyHlsToken(token);
248
+
249
+ if (!/^seg\d+\.ts$/.test(segment)) {
250
+ throw new NotFoundException('Invalid segment name');
251
+ }
252
+
253
+ const segmentFile = await (this.prisma as any).file.findFirst({
254
+ where: {
255
+ location: `lms/lessons/hls/${lessonId}/${variant}`,
256
+ filename: segment,
257
+ },
258
+ select: { id: true },
259
+ });
260
+
261
+ if (!segmentFile) throw new NotFoundException('HLS segment not found');
262
+
263
+ const { stream } = await this.fileService.openReadStreamById(segmentFile.id);
264
+
265
+ res.set({
266
+ 'Content-Type': 'video/MP2T',
267
+ 'Cache-Control': 'public, max-age=31536000, immutable',
268
+ 'Access-Control-Allow-Origin': '*',
269
+ });
270
+
271
+ stream.pipe(res);
272
+ }
273
+
274
+ // ---------------------------------------------------------------------------
275
+ // Subtitle (VTT) endpoints
276
+ // ---------------------------------------------------------------------------
277
+
278
+ async generateSubtitlesToken(lessonId: number) {
279
+ const lesson = await this.prisma.course_lesson.findFirst({
280
+ where: { id: lessonId, published: true },
281
+ select: { id: true },
282
+ });
283
+
284
+ if (!lesson) throw new NotFoundException('Lesson not found');
285
+
286
+ const count = await (this.prisma as any).course_lesson_transcription_segment.count({
287
+ where: { course_lesson_id: lessonId },
288
+ });
289
+
290
+ if (count === 0) throw new NotFoundException('No subtitles available for this lesson');
291
+
292
+ const token = this.jwtService.sign(
293
+ { lessonId, type: 'subtitles' },
294
+ { secret: process.env.JWT_SECRET, expiresIn: '4h' },
295
+ );
296
+
297
+ return { token, expiresIn: 14400 };
298
+ }
299
+
300
+ async serveSubtitles(token: string, res: Response) {
301
+ let lessonId: number;
302
+ try {
303
+ const payload = this.jwtService.verify<{ lessonId: number; type: string }>(token, {
304
+ secret: process.env.JWT_SECRET,
305
+ });
306
+ if (payload.type !== 'subtitles') throw new Error('Wrong token type');
307
+ lessonId = payload.lessonId;
308
+ } catch {
309
+ throw new UnauthorizedException('Invalid or expired subtitles token');
310
+ }
311
+
312
+ const segments = await (this.prisma as any).course_lesson_transcription_segment.findMany({
313
+ where: { course_lesson_id: lessonId },
314
+ orderBy: { start_seconds: 'asc' },
315
+ select: { start_seconds: true, end_seconds: true, text: true },
316
+ });
317
+
318
+ const vtt = this.buildVtt(segments);
319
+
320
+ res.set({
321
+ 'Content-Type': 'text/vtt; charset=utf-8',
322
+ 'Cache-Control': 'no-store',
323
+ 'Access-Control-Allow-Origin': '*',
324
+ });
325
+ res.send(vtt);
326
+ }
327
+
328
+ private secondsToVttTime(seconds: number): string {
329
+ const s = Number(seconds);
330
+ const h = Math.floor(s / 3600);
331
+ const m = Math.floor((s % 3600) / 60);
332
+ const sec = Math.floor(s % 60);
333
+ const ms = Math.round((s % 1) * 1000);
334
+ return `${String(h).padStart(2, '0')}:${String(m).padStart(2, '0')}:${String(sec).padStart(2, '0')}.${String(ms).padStart(3, '0')}`;
335
+ }
336
+
337
+ private buildVtt(segments: { start_seconds: any; end_seconds: any; text: string }[]): string {
338
+ if (segments.length === 0) return 'WEBVTT\n';
339
+ const cues = segments.map((seg, i) => {
340
+ const start = this.secondsToVttTime(seg.start_seconds);
341
+ const end = this.secondsToVttTime(seg.end_seconds);
342
+ return `${i + 1}\n${start} --> ${end}\n${seg.text}`;
343
+ });
344
+ return `WEBVTT\n\n${cues.join('\n\n')}\n`;
345
+ }
346
+ }
@@ -1,12 +1,18 @@
1
- import { NoRole, User } from '@hed-hog/api';
1
+ import { NoRole, Public, User } from '@hed-hog/api';
2
2
  import { NotificationService } from '@hed-hog/core';
3
3
  import { Body, Controller, Delete, Get, Param, ParseIntPipe, Patch, Post, Put, Query, Req, Res } from '@nestjs/common';
4
4
  import type { Request, Response } from 'express';
5
5
  import { BitcodeWalletService } from '../bitcode-wallet/bitcode-wallet.service';
6
6
  import { CreateCurrentBitcodeWalletTransactionDto } from '../bitcode-wallet/dto/create-current-bitcode-wallet-transaction.dto';
7
+ import { EvaluationService } from '../evaluation/evaluation.service';
7
8
  import { TrainingStudentService } from '../enterprise/training/training-student.service';
9
+ import { HeartbeatDto } from './dto/heartbeat.dto';
8
10
  import { UpdatePlatformaProfileDto } from './dto/update-profile.dto';
11
+ import { PlatformaHeartbeatService } from './platforma-heartbeat.service';
12
+ import { PlatformaPerformanceService } from './platforma-performance.service';
9
13
  import { PlatformaService } from './platforma.service';
14
+ import { PlatformaSearchService } from './platforma-search.service';
15
+ import { PlatformaVideoService } from './platforma-video.service';
10
16
 
11
17
  @NoRole()
12
18
  @Controller('lms/platforma')
@@ -16,6 +22,11 @@ export class PlataformaController {
16
22
  private readonly notificationService: NotificationService,
17
23
  private readonly platforma: PlatformaService,
18
24
  private readonly bitcodeWalletService: BitcodeWalletService,
25
+ private readonly platformaVideoService: PlatformaVideoService,
26
+ private readonly evaluationService: EvaluationService,
27
+ private readonly heartbeatService: PlatformaHeartbeatService,
28
+ private readonly performanceService: PlatformaPerformanceService,
29
+ private readonly searchService: PlatformaSearchService,
19
30
  ) {}
20
31
 
21
32
  @Get()
@@ -45,6 +56,109 @@ export class PlataformaController {
45
56
  });
46
57
  }
47
58
 
59
+ @Get('courses/:slug')
60
+ getCourseBySlug(@Param('slug') slug: string, @User('id') userId?: number) {
61
+ return this.trainingStudentService.getPublishedCourseBySlug(slug, userId);
62
+ }
63
+
64
+ @Get('search')
65
+ search(
66
+ @Query('q') q = '',
67
+ @Query('types') types?: string,
68
+ @Query('pageSize') pageSize?: string,
69
+ ) {
70
+ return this.searchService.search(
71
+ q,
72
+ types ? types.split(',').filter(Boolean) : [],
73
+ pageSize ? Math.max(1, Number(pageSize)) : 5,
74
+ );
75
+ }
76
+
77
+ @Get('lessons/:lessonId')
78
+ getLessonDetail(@Param('lessonId', ParseIntPipe) lessonId: number) {
79
+ return this.trainingStudentService.getLessonDetail(lessonId);
80
+ }
81
+
82
+ @Get('lessons/:lessonId/video-token')
83
+ getVideoToken(
84
+ @Param('lessonId', ParseIntPipe) lessonId: number,
85
+ @Query('quality') quality?: string,
86
+ ) {
87
+ return this.platformaVideoService.generateVideoToken(lessonId, quality);
88
+ }
89
+
90
+ @Get('lessons/:lessonId/hls-token')
91
+ getHlsToken(@Param('lessonId', ParseIntPipe) lessonId: number) {
92
+ return this.platformaVideoService.generateHlsToken(lessonId);
93
+ }
94
+
95
+ @Public()
96
+ @Get('video/:token')
97
+ streamVideo(@Param('token') token: string, @Req() req: Request, @Res() res: Response) {
98
+ return this.platformaVideoService.streamVideo(token, req, res);
99
+ }
100
+
101
+ @Public()
102
+ @Get('hls/:token/master.m3u8')
103
+ serveHlsMaster(@Param('token') token: string, @Req() req: Request, @Res() res: Response) {
104
+ return this.platformaVideoService.serveHlsMasterManifest(token, req, res);
105
+ }
106
+
107
+ @Public()
108
+ @Get('hls/:token/:variant/playlist.m3u8')
109
+ serveHlsVariant(
110
+ @Param('token') token: string,
111
+ @Param('variant') variant: string,
112
+ @Req() req: Request,
113
+ @Res() res: Response,
114
+ ) {
115
+ return this.platformaVideoService.serveHlsVariantPlaylist(token, variant, req, res);
116
+ }
117
+
118
+ @Public()
119
+ @Get('hls/:token/:variant/:segment')
120
+ serveHlsSegment(
121
+ @Param('token') token: string,
122
+ @Param('variant') variant: string,
123
+ @Param('segment') segment: string,
124
+ @Res() res: Response,
125
+ ) {
126
+ return this.platformaVideoService.serveHlsSegment(token, variant, segment, res);
127
+ }
128
+
129
+ @Get('lessons/:lessonId/subtitles-token')
130
+ getSubtitlesToken(@Param('lessonId', ParseIntPipe) lessonId: number) {
131
+ return this.platformaVideoService.generateSubtitlesToken(lessonId);
132
+ }
133
+
134
+ @Get('lessons/:lessonId/evaluation-topics')
135
+ getCourseLessonEvaluationTopics(@Param('lessonId', ParseIntPipe) lessonId: number) {
136
+ return this.evaluationService.getCourseLessonTopics(lessonId);
137
+ }
138
+
139
+ @Get('lessons/:lessonId/my-evaluation')
140
+ getMyLessonEvaluation(
141
+ @Param('lessonId', ParseIntPipe) lessonId: number,
142
+ @User('id') userId: number,
143
+ ) {
144
+ return this.evaluationService.getMyLessonRatings(lessonId, userId);
145
+ }
146
+
147
+ @Post('lessons/:lessonId/evaluate')
148
+ submitLessonEvaluation(
149
+ @Param('lessonId', ParseIntPipe) lessonId: number,
150
+ @User('id') userId: number,
151
+ @Body() body: { ratings: Array<{ topicId: number; score: number; comment?: string }> },
152
+ ) {
153
+ return this.evaluationService.submitLessonRatings(lessonId, userId, body.ratings ?? []);
154
+ }
155
+
156
+ @Public()
157
+ @Get('subtitles/:token/captions.vtt')
158
+ serveSubtitles(@Param('token') token: string, @Res() res: Response) {
159
+ return this.platformaVideoService.serveSubtitles(token, res);
160
+ }
161
+
48
162
  @Get('notifications')
49
163
  listNotifications(
50
164
  @User('id') userId: number,
@@ -132,6 +246,28 @@ export class PlataformaController {
132
246
  return this.bitcodeWalletService.createCurrentUserTransaction(userId, dto);
133
247
  }
134
248
 
249
+ @Get('performance/summary')
250
+ getPerformanceSummary(
251
+ @User('id') userId: number,
252
+ @Query('period') period?: string,
253
+ ) {
254
+ return this.performanceService.getSummary(userId, period ?? '30d');
255
+ }
256
+
257
+ @Post('heartbeat')
258
+ heartbeat(
259
+ @User('id') userId: number,
260
+ @Body() dto: HeartbeatDto,
261
+ @Req() req: Request,
262
+ ) {
263
+ const ip =
264
+ (req.headers['x-forwarded-for'] as string)?.split(',')[0]?.trim() ??
265
+ req.socket?.remoteAddress ??
266
+ '';
267
+ const userAgent = (req.headers['user-agent'] as string) ?? '';
268
+ return this.heartbeatService.enqueue(userId, ip, userAgent, dto);
269
+ }
270
+
135
271
  @Get('profile')
136
272
  getProfile(@User('id') userId: number) {
137
273
  return this.platforma.getProfile(userId);