@hed-hog/lms 0.0.358 → 0.0.364
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.
- package/dist/bitcode-wallet/bitcode-wallet.service.d.ts +65 -0
- package/dist/bitcode-wallet/bitcode-wallet.service.d.ts.map +1 -1
- package/dist/bitcode-wallet/bitcode-wallet.service.js +72 -0
- package/dist/bitcode-wallet/bitcode-wallet.service.js.map +1 -1
- package/dist/bitcode-wallet/dto/create-current-bitcode-wallet-transaction.dto.d.ts +8 -0
- package/dist/bitcode-wallet/dto/create-current-bitcode-wallet-transaction.dto.d.ts.map +1 -0
- package/dist/bitcode-wallet/dto/create-current-bitcode-wallet-transaction.dto.js +40 -0
- package/dist/bitcode-wallet/dto/create-current-bitcode-wallet-transaction.dto.js.map +1 -0
- package/dist/class-group/class-group.controller.d.ts +16 -16
- package/dist/class-group/class-group.service.d.ts +12 -12
- package/dist/course/course-audio-transcription.service.d.ts +3 -2
- package/dist/course/course-audio-transcription.service.d.ts.map +1 -1
- package/dist/course/course-audio-transcription.service.js +49 -8
- package/dist/course/course-audio-transcription.service.js.map +1 -1
- package/dist/course/course-lesson.controller.d.ts +4 -0
- package/dist/course/course-lesson.controller.d.ts.map +1 -1
- package/dist/course/course-lesson.controller.js +10 -0
- package/dist/course/course-lesson.controller.js.map +1 -1
- package/dist/course/course-structure.controller.d.ts +11 -4
- package/dist/course/course-structure.controller.d.ts.map +1 -1
- package/dist/course/course-structure.controller.js +14 -0
- package/dist/course/course-structure.controller.js.map +1 -1
- package/dist/course/course-structure.service.d.ts +15 -1
- package/dist/course/course-structure.service.d.ts.map +1 -1
- package/dist/course/course-structure.service.js +139 -4
- package/dist/course/course-structure.service.js.map +1 -1
- package/dist/course/course-video-conversion.service.d.ts +8 -0
- package/dist/course/course-video-conversion.service.d.ts.map +1 -1
- package/dist/course/course-video-conversion.service.js +87 -51
- package/dist/course/course-video-conversion.service.js.map +1 -1
- package/dist/course/course.controller.d.ts +73 -1
- package/dist/course/course.controller.d.ts.map +1 -1
- package/dist/course/course.controller.js +27 -3
- package/dist/course/course.controller.js.map +1 -1
- package/dist/course/course.module.d.ts.map +1 -1
- package/dist/course/course.module.js +15 -3
- package/dist/course/course.module.js.map +1 -1
- package/dist/course/course.service.d.ts +108 -4
- package/dist/course/course.service.d.ts.map +1 -1
- package/dist/course/course.service.js +631 -30
- package/dist/course/course.service.js.map +1 -1
- package/dist/course/dto/cleanup-course-storage.dto.d.ts +6 -0
- package/dist/course/dto/cleanup-course-storage.dto.d.ts.map +1 -0
- package/dist/course/dto/cleanup-course-storage.dto.js +33 -0
- package/dist/course/dto/cleanup-course-storage.dto.js.map +1 -0
- package/dist/course/dto/cleanup-upload-history.dto.d.ts +9 -0
- package/dist/course/dto/cleanup-upload-history.dto.d.ts.map +1 -0
- package/dist/course/dto/cleanup-upload-history.dto.js +36 -0
- package/dist/course/dto/cleanup-upload-history.dto.js.map +1 -0
- package/dist/course/dto/create-course-bulk-job.dto.d.ts +4 -0
- package/dist/course/dto/create-course-bulk-job.dto.d.ts.map +1 -0
- package/dist/course/dto/create-course-bulk-job.dto.js +21 -0
- package/dist/course/dto/create-course-bulk-job.dto.js.map +1 -0
- package/dist/course/lms-bulk-upload-automation.service.d.ts +39 -0
- package/dist/course/lms-bulk-upload-automation.service.d.ts.map +1 -0
- package/dist/course/lms-bulk-upload-automation.service.js +443 -0
- package/dist/course/lms-bulk-upload-automation.service.js.map +1 -0
- package/dist/course/lms-bulk-upload-infra.service.d.ts +31 -0
- package/dist/course/lms-bulk-upload-infra.service.d.ts.map +1 -0
- package/dist/course/lms-bulk-upload-infra.service.js +277 -0
- package/dist/course/lms-bulk-upload-infra.service.js.map +1 -0
- package/dist/course/lms-bulk-upload.constants.d.ts +4 -0
- package/dist/course/lms-bulk-upload.constants.d.ts.map +1 -0
- package/dist/course/lms-bulk-upload.constants.js +7 -0
- package/dist/course/lms-bulk-upload.constants.js.map +1 -0
- package/dist/course/lms-bulk-upload.controller.d.ts +153 -0
- package/dist/course/lms-bulk-upload.controller.d.ts.map +1 -0
- package/dist/course/lms-bulk-upload.controller.js +129 -0
- package/dist/course/lms-bulk-upload.controller.js.map +1 -0
- package/dist/course/lms-bulk-upload.service.d.ts +181 -0
- package/dist/course/lms-bulk-upload.service.d.ts.map +1 -0
- package/dist/course/lms-bulk-upload.service.js +754 -0
- package/dist/course/lms-bulk-upload.service.js.map +1 -0
- package/dist/course/lms-setting.controller.d.ts +7 -1
- package/dist/course/lms-setting.controller.d.ts.map +1 -1
- package/dist/course/lms-setting.controller.js +26 -3
- package/dist/course/lms-setting.controller.js.map +1 -1
- package/dist/enterprise/enterprise.controller.d.ts +20 -20
- package/dist/enterprise/enterprise.service.d.ts +20 -20
- package/dist/enterprise/training/training-admin.controller.d.ts +11 -11
- package/dist/enterprise/training/training-admin.service.d.ts +11 -11
- package/dist/enterprise/training/training-instructor.controller.d.ts +2 -2
- package/dist/enterprise/training/training-instructor.service.d.ts +2 -2
- package/dist/enterprise/training/training-student.controller.d.ts +1 -1
- package/dist/enterprise/training/training-student.service.d.ts +1 -1
- package/dist/enterprise/training/training-viewer.controller.d.ts +2 -2
- package/dist/evaluation/evaluation.controller.d.ts +8 -8
- package/dist/evaluation/evaluation.service.d.ts +8 -8
- package/dist/lesson-xp-map/dto/create-lesson-xp-map.dto.d.ts +6 -0
- package/dist/lesson-xp-map/dto/create-lesson-xp-map.dto.d.ts.map +1 -0
- package/dist/lesson-xp-map/dto/create-lesson-xp-map.dto.js +34 -0
- package/dist/lesson-xp-map/dto/create-lesson-xp-map.dto.js.map +1 -0
- package/dist/lesson-xp-map/dto/create-lesson-xp-segment.dto.d.ts +28 -0
- package/dist/lesson-xp-map/dto/create-lesson-xp-segment.dto.d.ts.map +1 -0
- package/dist/lesson-xp-map/dto/create-lesson-xp-segment.dto.js +123 -0
- package/dist/lesson-xp-map/dto/create-lesson-xp-segment.dto.js.map +1 -0
- package/dist/lesson-xp-map/dto/review-lesson-xp-map.dto.d.ts +4 -0
- package/dist/lesson-xp-map/dto/review-lesson-xp-map.dto.d.ts.map +1 -0
- package/dist/lesson-xp-map/dto/review-lesson-xp-map.dto.js +22 -0
- package/dist/lesson-xp-map/dto/review-lesson-xp-map.dto.js.map +1 -0
- package/dist/lesson-xp-map/dto/update-lesson-xp-map.dto.d.ts +10 -0
- package/dist/lesson-xp-map/dto/update-lesson-xp-map.dto.d.ts.map +1 -0
- package/dist/lesson-xp-map/dto/update-lesson-xp-map.dto.js +52 -0
- package/dist/lesson-xp-map/dto/update-lesson-xp-map.dto.js.map +1 -0
- package/dist/lesson-xp-map/dto/update-lesson-xp-segment.dto.d.ts +15 -0
- package/dist/lesson-xp-map/dto/update-lesson-xp-segment.dto.d.ts.map +1 -0
- package/dist/lesson-xp-map/dto/update-lesson-xp-segment.dto.js +86 -0
- package/dist/lesson-xp-map/dto/update-lesson-xp-segment.dto.js.map +1 -0
- package/dist/lesson-xp-map/lesson-xp-ai-calculation.service.d.ts +26 -0
- package/dist/lesson-xp-map/lesson-xp-ai-calculation.service.d.ts.map +1 -0
- package/dist/lesson-xp-map/lesson-xp-ai-calculation.service.js +304 -0
- package/dist/lesson-xp-map/lesson-xp-ai-calculation.service.js.map +1 -0
- package/dist/lesson-xp-map/lesson-xp-map.controller.d.ts +87 -0
- package/dist/lesson-xp-map/lesson-xp-map.controller.d.ts.map +1 -0
- package/dist/lesson-xp-map/lesson-xp-map.controller.js +185 -0
- package/dist/lesson-xp-map/lesson-xp-map.controller.js.map +1 -0
- package/dist/lesson-xp-map/lesson-xp-map.module.d.ts +3 -0
- package/dist/lesson-xp-map/lesson-xp-map.module.d.ts.map +1 -0
- package/dist/lesson-xp-map/lesson-xp-map.module.js +34 -0
- package/dist/lesson-xp-map/lesson-xp-map.module.js.map +1 -0
- package/dist/lesson-xp-map/lesson-xp-map.service.d.ts +84 -0
- package/dist/lesson-xp-map/lesson-xp-map.service.d.ts.map +1 -0
- package/dist/lesson-xp-map/lesson-xp-map.service.js +353 -0
- package/dist/lesson-xp-map/lesson-xp-map.service.js.map +1 -0
- package/dist/lesson-xp-map/lesson-xp-segment.controller.d.ts +10 -0
- package/dist/lesson-xp-map/lesson-xp-segment.controller.d.ts.map +1 -0
- package/dist/lesson-xp-map/lesson-xp-segment.controller.js +63 -0
- package/dist/lesson-xp-map/lesson-xp-segment.controller.js.map +1 -0
- package/dist/lesson-xp-map/lesson-xp-segment.service.d.ts +27 -0
- package/dist/lesson-xp-map/lesson-xp-segment.service.d.ts.map +1 -0
- package/dist/lesson-xp-map/lesson-xp-segment.service.js +194 -0
- package/dist/lesson-xp-map/lesson-xp-segment.service.js.map +1 -0
- package/dist/libraries/lms/tsconfig.tsbuildinfo +1 -0
- package/dist/lms.module.d.ts.map +1 -1
- package/dist/lms.module.js +17 -2
- package/dist/lms.module.js.map +1 -1
- package/dist/platforma/dto/update-profile.dto.d.ts +17 -0
- package/dist/platforma/dto/update-profile.dto.d.ts.map +1 -0
- package/dist/platforma/dto/update-profile.dto.js +87 -0
- package/dist/platforma/dto/update-profile.dto.js.map +1 -0
- package/dist/platforma/platforma.controller.d.ts +94 -7
- package/dist/platforma/platforma.controller.d.ts.map +1 -1
- package/dist/platforma/platforma.controller.js +85 -2
- package/dist/platforma/platforma.controller.js.map +1 -1
- package/dist/platforma/platforma.service.d.ts +27 -0
- package/dist/platforma/platforma.service.d.ts.map +1 -0
- package/dist/platforma/platforma.service.js +274 -0
- package/dist/platforma/platforma.service.js.map +1 -0
- package/dist/student-xp/student-xp.controller.d.ts +41 -0
- package/dist/student-xp/student-xp.controller.d.ts.map +1 -0
- package/dist/student-xp/student-xp.controller.js +114 -0
- package/dist/student-xp/student-xp.controller.js.map +1 -0
- package/dist/student-xp/student-xp.module.d.ts +3 -0
- package/dist/student-xp/student-xp.module.d.ts.map +1 -0
- package/dist/student-xp/student-xp.module.js +25 -0
- package/dist/student-xp/student-xp.module.js.map +1 -0
- package/dist/student-xp/student-xp.service.d.ts +65 -0
- package/dist/student-xp/student-xp.service.d.ts.map +1 -0
- package/dist/student-xp/student-xp.service.js +197 -0
- package/dist/student-xp/student-xp.service.js.map +1 -0
- package/dist/xp-catalog/dto/create-xp-area.dto.d.ts +12 -0
- package/dist/xp-catalog/dto/create-xp-area.dto.d.ts.map +1 -0
- package/dist/xp-catalog/dto/create-xp-area.dto.js +63 -0
- package/dist/xp-catalog/dto/create-xp-area.dto.js.map +1 -0
- package/dist/xp-catalog/dto/create-xp-learning-type.dto.d.ts +11 -0
- package/dist/xp-catalog/dto/create-xp-learning-type.dto.d.ts.map +1 -0
- package/dist/xp-catalog/dto/create-xp-learning-type.dto.js +57 -0
- package/dist/xp-catalog/dto/create-xp-learning-type.dto.js.map +1 -0
- package/dist/xp-catalog/dto/create-xp-skill.dto.d.ts +11 -0
- package/dist/xp-catalog/dto/create-xp-skill.dto.d.ts.map +1 -0
- package/dist/xp-catalog/dto/create-xp-skill.dto.js +57 -0
- package/dist/xp-catalog/dto/create-xp-skill.dto.js.map +1 -0
- package/dist/xp-catalog/dto/update-xp-area.dto.d.ts +12 -0
- package/dist/xp-catalog/dto/update-xp-area.dto.d.ts.map +1 -0
- package/dist/xp-catalog/dto/update-xp-area.dto.js +66 -0
- package/dist/xp-catalog/dto/update-xp-area.dto.js.map +1 -0
- package/dist/xp-catalog/dto/update-xp-learning-type.dto.d.ts +11 -0
- package/dist/xp-catalog/dto/update-xp-learning-type.dto.d.ts.map +1 -0
- package/dist/xp-catalog/dto/update-xp-learning-type.dto.js +60 -0
- package/dist/xp-catalog/dto/update-xp-learning-type.dto.js.map +1 -0
- package/dist/xp-catalog/dto/update-xp-skill.dto.d.ts +11 -0
- package/dist/xp-catalog/dto/update-xp-skill.dto.d.ts.map +1 -0
- package/dist/xp-catalog/dto/update-xp-skill.dto.js +60 -0
- package/dist/xp-catalog/dto/update-xp-skill.dto.js.map +1 -0
- package/dist/xp-catalog/xp-area.controller.d.ts +25 -0
- package/dist/xp-catalog/xp-area.controller.d.ts.map +1 -0
- package/dist/xp-catalog/xp-area.controller.js +105 -0
- package/dist/xp-catalog/xp-area.controller.js.map +1 -0
- package/dist/xp-catalog/xp-area.service.d.ts +35 -0
- package/dist/xp-catalog/xp-area.service.d.ts.map +1 -0
- package/dist/xp-catalog/xp-area.service.js +168 -0
- package/dist/xp-catalog/xp-area.service.js.map +1 -0
- package/dist/xp-catalog/xp-catalog.module.d.ts +3 -0
- package/dist/xp-catalog/xp-catalog.module.d.ts.map +1 -0
- package/dist/xp-catalog/xp-catalog.module.js +29 -0
- package/dist/xp-catalog/xp-catalog.module.js.map +1 -0
- package/dist/xp-catalog/xp-learning-type.controller.d.ts +20 -0
- package/dist/xp-catalog/xp-learning-type.controller.d.ts.map +1 -0
- package/dist/xp-catalog/xp-learning-type.controller.js +96 -0
- package/dist/xp-catalog/xp-learning-type.controller.js.map +1 -0
- package/dist/xp-catalog/xp-learning-type.service.d.ts +30 -0
- package/dist/xp-catalog/xp-learning-type.service.d.ts.map +1 -0
- package/dist/xp-catalog/xp-learning-type.service.js +146 -0
- package/dist/xp-catalog/xp-learning-type.service.js.map +1 -0
- package/dist/xp-catalog/xp-skill.controller.d.ts +26 -0
- package/dist/xp-catalog/xp-skill.controller.d.ts.map +1 -0
- package/dist/xp-catalog/xp-skill.controller.js +113 -0
- package/dist/xp-catalog/xp-skill.controller.js.map +1 -0
- package/dist/xp-catalog/xp-skill.service.d.ts +37 -0
- package/dist/xp-catalog/xp-skill.service.d.ts.map +1 -0
- package/dist/xp-catalog/xp-skill.service.js +174 -0
- package/dist/xp-catalog/xp-skill.service.js.map +1 -0
- package/hedhog/data/menu.yaml +101 -0
- package/hedhog/data/role.yaml +8 -0
- package/hedhog/data/route.yaml +547 -0
- package/hedhog/data/setting_group.yaml +33 -0
- package/hedhog/data/xp_area.yaml +164 -0
- package/hedhog/data/xp_learning_type.yaml +131 -0
- package/hedhog/data/xp_skill.yaml +1834 -0
- package/hedhog/frontend/app/achievements/page.tsx.ejs +108 -118
- package/hedhog/frontend/app/bitcodes/page.tsx.ejs +22 -34
- package/hedhog/frontend/app/bulk-upload-sessions/page.tsx.ejs +1453 -0
- package/hedhog/frontend/app/certificates/models/page.tsx.ejs +21 -45
- package/hedhog/frontend/app/classes/[id]/page.tsx.ejs +40 -74
- package/hedhog/frontend/app/classes/page.tsx.ejs +56 -85
- package/hedhog/frontend/app/courses/[id]/_components/CourseDangerZoneCard.tsx.ejs +3 -2
- package/hedhog/frontend/app/courses/[id]/_components/CourseMediaCard.tsx.ejs +48 -5
- package/hedhog/frontend/app/courses/[id]/page.tsx.ejs +4 -4
- package/hedhog/frontend/app/courses/[id]/structure/_components/course-operations-tab.tsx.ejs +19 -2
- package/hedhog/frontend/app/courses/[id]/structure/_components/course-overview-tab.tsx.ejs +1170 -0
- package/hedhog/frontend/app/courses/[id]/structure/_components/course-tree-dnd.tsx.ejs +16 -0
- package/hedhog/frontend/app/courses/[id]/structure/_components/course-tree.tsx.ejs +2 -0
- package/hedhog/frontend/app/courses/[id]/structure/_components/course-xp-overview-tab.tsx.ejs +623 -0
- package/hedhog/frontend/app/courses/[id]/structure/_components/detail-lesson-xp-tab.tsx.ejs +1458 -0
- package/hedhog/frontend/app/courses/[id]/structure/_components/detail-lesson.tsx.ejs +55 -2
- package/hedhog/frontend/app/courses/[id]/structure/_components/editor-course.tsx.ejs +442 -104
- package/hedhog/frontend/app/courses/[id]/structure/_components/editor-lesson.tsx.ejs +296 -49
- package/hedhog/frontend/app/courses/[id]/structure/_components/sortable-tree-row.tsx.ejs +3 -0
- package/hedhog/frontend/app/courses/[id]/structure/_components/store.ts.ejs +1 -0
- package/hedhog/frontend/app/courses/[id]/structure/_components/tree-display-settings-popover.tsx.ejs +101 -85
- package/hedhog/frontend/app/courses/[id]/structure/_components/tree-row-lesson.tsx.ejs +21 -1
- package/hedhog/frontend/app/courses/[id]/structure/_components/tree-row.tsx.ejs +3 -0
- package/hedhog/frontend/app/courses/[id]/structure/_components/use-tree-display-settings.ts.ejs +7 -1
- package/hedhog/frontend/app/courses/[id]/structure/_components/xp-premium-pills.tsx.ejs +44 -0
- package/hedhog/frontend/app/courses/[id]/structure/_data/use-course-content-overview.ts.ejs +54 -0
- package/hedhog/frontend/app/courses/[id]/structure/_data/use-course-structure-mutations.ts.ejs +52 -0
- package/hedhog/frontend/app/courses/[id]/structure/_data/use-course-xp-overview.ts.ejs +76 -0
- package/hedhog/frontend/app/courses/[id]/structure/_data/use-lesson-xp-map.ts.ejs +128 -0
- package/hedhog/frontend/app/courses/[id]/structure/_data/use-transcription-segments.ts.ejs +30 -0
- package/hedhog/frontend/app/courses/[id]/structure/_utils/xp-color-config.ts.ejs +115 -0
- package/hedhog/frontend/app/courses/_components/CourseDeleteDialog.tsx.ejs +223 -0
- package/hedhog/frontend/app/courses/_components/CourseRowActions.tsx.ejs +89 -0
- package/hedhog/frontend/app/courses/page.tsx.ejs +400 -230
- package/hedhog/frontend/app/enterprise/page.tsx.ejs +39 -63
- package/hedhog/frontend/app/exams/[id]/questions/page.tsx.ejs +53 -77
- package/hedhog/frontend/app/exams/page.tsx.ejs +54 -90
- package/hedhog/frontend/app/instructor-skills/page.tsx.ejs +23 -36
- package/hedhog/frontend/app/instructors/page.tsx.ejs +72 -81
- package/hedhog/frontend/app/paths/page.tsx.ejs +40 -68
- package/hedhog/frontend/app/training/page.tsx.ejs +40 -68
- package/hedhog/frontend/app/xp/areas/page.tsx.ejs +782 -0
- package/hedhog/frontend/app/xp/learning-types/page.tsx.ejs +690 -0
- package/hedhog/frontend/app/xp/skills/page.tsx.ejs +811 -0
- package/hedhog/frontend/messages/en.json +386 -3
- package/hedhog/frontend/messages/pt.json +386 -3
- package/hedhog/table/lesson_xp_map.yaml +50 -0
- package/hedhog/table/lesson_xp_segment.yaml +40 -0
- package/hedhog/table/lesson_xp_segment_area.yaml +24 -0
- package/hedhog/table/lesson_xp_segment_learning_type.yaml +24 -0
- package/hedhog/table/lesson_xp_segment_skill.yaml +24 -0
- package/hedhog/table/lms_bulk_upload_item.yaml +44 -0
- package/hedhog/table/lms_bulk_upload_session.yaml +42 -0
- package/hedhog/table/student_area_xp.yaml +30 -0
- package/hedhog/table/student_learning_type_xp.yaml +30 -0
- package/hedhog/table/student_skill_xp.yaml +30 -0
- package/hedhog/table/student_xp_event.yaml +34 -0
- package/hedhog/table/xp_area.yaml +39 -0
- package/hedhog/table/xp_learning_type.yaml +34 -0
- package/hedhog/table/xp_skill.yaml +39 -0
- package/package.json +9 -8
- package/src/bitcode-wallet/bitcode-wallet.service.ts +113 -0
- package/src/bitcode-wallet/dto/create-current-bitcode-wallet-transaction.dto.ts +32 -0
- package/src/course/course-audio-transcription.service.ts +58 -21
- package/src/course/course-lesson.controller.ts +6 -1
- package/src/course/course-structure.controller.ts +10 -0
- package/src/course/course-structure.service.ts +174 -1
- package/src/course/course-video-conversion.service.ts +113 -75
- package/src/course/course.controller.ts +22 -3
- package/src/course/course.module.ts +15 -3
- package/src/course/course.service.ts +847 -30
- package/src/course/dto/cleanup-course-storage.dto.ts +22 -0
- package/src/course/dto/cleanup-upload-history.dto.ts +26 -0
- package/src/course/dto/create-course-bulk-job.dto.ts +6 -0
- package/src/course/lms-bulk-upload-automation.service.ts +560 -0
- package/src/course/lms-bulk-upload-infra.service.ts +327 -0
- package/src/course/lms-bulk-upload.constants.ts +5 -0
- package/src/course/lms-bulk-upload.controller.ts +103 -0
- package/src/course/lms-bulk-upload.service.ts +1029 -0
- package/src/course/lms-setting.controller.ts +28 -1
- package/src/lesson-xp-map/dto/create-lesson-xp-map.dto.ts +17 -0
- package/src/lesson-xp-map/dto/create-lesson-xp-segment.dto.ts +102 -0
- package/src/lesson-xp-map/dto/review-lesson-xp-map.dto.ts +7 -0
- package/src/lesson-xp-map/dto/update-lesson-xp-map.dto.ts +36 -0
- package/src/lesson-xp-map/dto/update-lesson-xp-segment.dto.ts +78 -0
- package/src/lesson-xp-map/lesson-xp-ai-calculation.service.ts +396 -0
- package/src/lesson-xp-map/lesson-xp-map.controller.ts +116 -0
- package/src/lesson-xp-map/lesson-xp-map.module.ts +21 -0
- package/src/lesson-xp-map/lesson-xp-map.service.ts +442 -0
- package/src/lesson-xp-map/lesson-xp-segment.controller.ts +36 -0
- package/src/lesson-xp-map/lesson-xp-segment.service.ts +229 -0
- package/src/lms.module.ts +17 -2
- package/src/platforma/dto/update-profile.dto.ts +59 -0
- package/src/platforma/platforma.controller.ts +57 -2
- package/src/platforma/platforma.service.ts +268 -0
- package/src/student-xp/student-xp.controller.ts +76 -0
- package/src/student-xp/student-xp.module.ts +12 -0
- package/src/student-xp/student-xp.service.ts +236 -0
- package/src/xp-catalog/dto/create-xp-area.dto.ts +40 -0
- package/src/xp-catalog/dto/create-xp-learning-type.dto.ts +35 -0
- package/src/xp-catalog/dto/create-xp-skill.dto.ts +35 -0
- package/src/xp-catalog/dto/update-xp-area.dto.ts +43 -0
- package/src/xp-catalog/dto/update-xp-learning-type.dto.ts +38 -0
- package/src/xp-catalog/dto/update-xp-skill.dto.ts +38 -0
- package/src/xp-catalog/xp-area.controller.ts +64 -0
- package/src/xp-catalog/xp-area.service.ts +196 -0
- package/src/xp-catalog/xp-catalog.module.ts +16 -0
- package/src/xp-catalog/xp-learning-type.controller.ts +59 -0
- package/src/xp-catalog/xp-learning-type.service.ts +170 -0
- package/src/xp-catalog/xp-skill.controller.ts +71 -0
- package/src/xp-catalog/xp-skill.service.ts +205 -0
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
import { IsIn, IsString } from 'class-validator';
|
|
2
|
+
|
|
3
|
+
export const COURSE_STORAGE_CLEANUP_CATEGORIES = [
|
|
4
|
+
'video_original',
|
|
5
|
+
'video_profile',
|
|
6
|
+
'lesson_audio',
|
|
7
|
+
'extracted_image',
|
|
8
|
+
'student_download',
|
|
9
|
+
'supplementary_material',
|
|
10
|
+
'course_image',
|
|
11
|
+
'course_file',
|
|
12
|
+
'other_lesson_file',
|
|
13
|
+
] as const;
|
|
14
|
+
|
|
15
|
+
export type CourseStorageCleanupCategory =
|
|
16
|
+
(typeof COURSE_STORAGE_CLEANUP_CATEGORIES)[number];
|
|
17
|
+
|
|
18
|
+
export class CleanupCourseStorageDto {
|
|
19
|
+
@IsString()
|
|
20
|
+
@IsIn(COURSE_STORAGE_CLEANUP_CATEGORIES)
|
|
21
|
+
category!: CourseStorageCleanupCategory;
|
|
22
|
+
}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
import { ArrayNotEmpty, IsArray, IsIn, IsOptional } from 'class-validator';
|
|
2
|
+
|
|
3
|
+
export const BULK_UPLOAD_CLEANUP_STATUSES = ['done', 'error', 'cancelled'] as const;
|
|
4
|
+
export const BULK_UPLOAD_CLEANUP_WINDOWS = [
|
|
5
|
+
'last_hour',
|
|
6
|
+
'last_day',
|
|
7
|
+
'last_week',
|
|
8
|
+
'all_time',
|
|
9
|
+
] as const;
|
|
10
|
+
|
|
11
|
+
export type BulkUploadCleanupStatus =
|
|
12
|
+
(typeof BULK_UPLOAD_CLEANUP_STATUSES)[number];
|
|
13
|
+
export type BulkUploadCleanupWindow =
|
|
14
|
+
(typeof BULK_UPLOAD_CLEANUP_WINDOWS)[number];
|
|
15
|
+
|
|
16
|
+
export class CleanupUploadHistoryDto {
|
|
17
|
+
@IsOptional()
|
|
18
|
+
@IsArray()
|
|
19
|
+
@ArrayNotEmpty()
|
|
20
|
+
@IsIn(BULK_UPLOAD_CLEANUP_STATUSES, { each: true })
|
|
21
|
+
statuses?: BulkUploadCleanupStatus[];
|
|
22
|
+
|
|
23
|
+
@IsOptional()
|
|
24
|
+
@IsIn(BULK_UPLOAD_CLEANUP_WINDOWS)
|
|
25
|
+
timeWindow?: BulkUploadCleanupWindow;
|
|
26
|
+
}
|
|
@@ -0,0 +1,560 @@
|
|
|
1
|
+
import { GetObjectCommand, S3Client } from '@aws-sdk/client-s3';
|
|
2
|
+
import { PrismaService } from '@hed-hog/api-prisma';
|
|
3
|
+
import {
|
|
4
|
+
FileService,
|
|
5
|
+
SettingService,
|
|
6
|
+
WebhookCommandRegistry,
|
|
7
|
+
WebhookIntegrationService,
|
|
8
|
+
} from '@hed-hog/core';
|
|
9
|
+
import { QueueHandlerRegistry, QueueJobService } from '@hed-hog/queue';
|
|
10
|
+
import {
|
|
11
|
+
BadRequestException,
|
|
12
|
+
Inject,
|
|
13
|
+
Injectable,
|
|
14
|
+
Logger,
|
|
15
|
+
OnModuleInit,
|
|
16
|
+
forwardRef,
|
|
17
|
+
} from '@nestjs/common';
|
|
18
|
+
import { createWriteStream, promises as fs } from 'fs';
|
|
19
|
+
import { tmpdir } from 'os';
|
|
20
|
+
import { basename, extname, join } from 'path';
|
|
21
|
+
import { Readable } from 'stream';
|
|
22
|
+
import { pipeline } from 'stream/promises';
|
|
23
|
+
import { CourseVideoConversionService } from './course-video-conversion.service';
|
|
24
|
+
import {
|
|
25
|
+
LMS_BULK_UPLOAD_DOWNLOAD_ORIGINAL_JOB,
|
|
26
|
+
LMS_BULK_UPLOAD_RECEIVE_VIDEO_COMMAND,
|
|
27
|
+
} from './lms-bulk-upload.constants';
|
|
28
|
+
|
|
29
|
+
type StorageContext = {
|
|
30
|
+
region: string;
|
|
31
|
+
bucket: string;
|
|
32
|
+
accessKeyId: string;
|
|
33
|
+
secretAccessKey: string;
|
|
34
|
+
sessionToken?: string;
|
|
35
|
+
roleArn?: string;
|
|
36
|
+
externalId?: string;
|
|
37
|
+
durationSeconds: number;
|
|
38
|
+
};
|
|
39
|
+
|
|
40
|
+
@Injectable()
|
|
41
|
+
export class LmsBulkUploadAutomationService implements OnModuleInit {
|
|
42
|
+
private readonly logger = new Logger(LmsBulkUploadAutomationService.name);
|
|
43
|
+
|
|
44
|
+
constructor(
|
|
45
|
+
@Inject(forwardRef(() => PrismaService))
|
|
46
|
+
private readonly prisma: PrismaService,
|
|
47
|
+
@Inject(forwardRef(() => SettingService))
|
|
48
|
+
private readonly settingService: SettingService,
|
|
49
|
+
@Inject(forwardRef(() => FileService))
|
|
50
|
+
private readonly fileService: FileService,
|
|
51
|
+
@Inject(forwardRef(() => QueueHandlerRegistry))
|
|
52
|
+
private readonly queueHandlers: QueueHandlerRegistry,
|
|
53
|
+
@Inject(forwardRef(() => QueueJobService))
|
|
54
|
+
private readonly queueJobService: QueueJobService,
|
|
55
|
+
@Inject(forwardRef(() => WebhookCommandRegistry))
|
|
56
|
+
private readonly commandRegistry: WebhookCommandRegistry,
|
|
57
|
+
@Inject(forwardRef(() => WebhookIntegrationService))
|
|
58
|
+
private readonly webhookIntegrationService: WebhookIntegrationService,
|
|
59
|
+
@Inject(forwardRef(() => CourseVideoConversionService))
|
|
60
|
+
private readonly courseVideoConversionService: CourseVideoConversionService,
|
|
61
|
+
) {}
|
|
62
|
+
|
|
63
|
+
onModuleInit(): void {
|
|
64
|
+
this.queueHandlers.register(LMS_BULK_UPLOAD_DOWNLOAD_ORIGINAL_JOB, this as any);
|
|
65
|
+
this.commandRegistry.register({
|
|
66
|
+
slug: LMS_BULK_UPLOAD_RECEIVE_VIDEO_COMMAND,
|
|
67
|
+
name: 'Receber vídeo do upload em massa',
|
|
68
|
+
description:
|
|
69
|
+
'Reconhece o arquivo recebido no upload em massa, vincula a aula e agenda o download do vídeo original do S3.',
|
|
70
|
+
inputSchema: {
|
|
71
|
+
type: 'object',
|
|
72
|
+
properties: {
|
|
73
|
+
bucket: { type: 'string' },
|
|
74
|
+
key: { type: 'string' },
|
|
75
|
+
fileName: { type: 'string' },
|
|
76
|
+
uploadId: { type: 'string' },
|
|
77
|
+
},
|
|
78
|
+
additionalProperties: true,
|
|
79
|
+
},
|
|
80
|
+
handler: async (_params, context) => this.handleWebhookReceive(context ?? {}),
|
|
81
|
+
});
|
|
82
|
+
|
|
83
|
+
this.logger.log(
|
|
84
|
+
`Registered webhook command and queue handler for LMS bulk upload automation`,
|
|
85
|
+
);
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
private async handleWebhookReceive(context: Record<string, unknown>) {
|
|
89
|
+
const payload = this.normalizeWebhookBody(context.webhookBody ?? context);
|
|
90
|
+
const bucket = payload.bucket || '';
|
|
91
|
+
const key = payload.key || '';
|
|
92
|
+
const fileName = payload.fileName || basename(key);
|
|
93
|
+
|
|
94
|
+
if (!bucket || !key) {
|
|
95
|
+
throw new BadRequestException('Webhook payload does not contain bucket and key.');
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
const match = await this.findLessonMatch(fileName);
|
|
99
|
+
const uploadItem = await this.findUploadItem(payload.uploadId, fileName, key);
|
|
100
|
+
const now = new Date();
|
|
101
|
+
|
|
102
|
+
if (uploadItem) {
|
|
103
|
+
await this.prisma.$executeRawUnsafe(
|
|
104
|
+
`UPDATE "lms_bulk_upload_item"
|
|
105
|
+
SET
|
|
106
|
+
"status" = 'received',
|
|
107
|
+
"progress_percent" = 100,
|
|
108
|
+
"error_message" = NULL,
|
|
109
|
+
"uploaded_key" = $2,
|
|
110
|
+
"received_at" = $3,
|
|
111
|
+
"completed_at" = $3,
|
|
112
|
+
"matched_course_id" = $4,
|
|
113
|
+
"matched_session_id" = $5,
|
|
114
|
+
"matched_lesson_id" = $6,
|
|
115
|
+
"updated_at" = $3
|
|
116
|
+
WHERE "id" = $1`,
|
|
117
|
+
uploadItem.id,
|
|
118
|
+
key,
|
|
119
|
+
now,
|
|
120
|
+
match?.courseId ?? null,
|
|
121
|
+
match?.sessionId ?? null,
|
|
122
|
+
match?.lessonId ?? null,
|
|
123
|
+
);
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
if (!match) {
|
|
127
|
+
return {
|
|
128
|
+
received: true,
|
|
129
|
+
matched: false,
|
|
130
|
+
uploadItemId: uploadItem?.id ?? null,
|
|
131
|
+
};
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
const job = await this.queueJobService.enqueue({
|
|
135
|
+
type: LMS_BULK_UPLOAD_DOWNLOAD_ORIGINAL_JOB,
|
|
136
|
+
queueName: LMS_BULK_UPLOAD_DOWNLOAD_ORIGINAL_JOB,
|
|
137
|
+
payload: {
|
|
138
|
+
bucket,
|
|
139
|
+
key,
|
|
140
|
+
fileName,
|
|
141
|
+
uploadItemId: uploadItem?.id ?? null,
|
|
142
|
+
userId: uploadItem?.userId ?? payload.userId ?? 0,
|
|
143
|
+
courseId: match.courseId,
|
|
144
|
+
sessionId: match.sessionId,
|
|
145
|
+
lessonId: match.lessonId,
|
|
146
|
+
matchedCourseId: match.courseId,
|
|
147
|
+
matchedSessionId: match.sessionId,
|
|
148
|
+
matchedLessonId: match.lessonId,
|
|
149
|
+
},
|
|
150
|
+
maxAttempts: 3,
|
|
151
|
+
sourceModule: 'lms',
|
|
152
|
+
sourceEntity: 'lms_bulk_upload_item',
|
|
153
|
+
sourceEntityId: uploadItem?.id ? String(uploadItem.id) : fileName,
|
|
154
|
+
priority: -6,
|
|
155
|
+
}, uploadItem?.userId ?? payload.userId ?? undefined);
|
|
156
|
+
|
|
157
|
+
return {
|
|
158
|
+
received: true,
|
|
159
|
+
matched: true,
|
|
160
|
+
uploadItemId: uploadItem?.id ?? null,
|
|
161
|
+
queueJobId: job.id,
|
|
162
|
+
lesson: {
|
|
163
|
+
courseId: match.courseId,
|
|
164
|
+
sessionId: match.sessionId,
|
|
165
|
+
lessonId: match.lessonId,
|
|
166
|
+
courseTitle: match.courseTitle,
|
|
167
|
+
sessionTitle: match.sessionTitle,
|
|
168
|
+
lessonTitle: match.lessonTitle,
|
|
169
|
+
courseLogoFileId: match.courseLogoFileId,
|
|
170
|
+
},
|
|
171
|
+
};
|
|
172
|
+
}
|
|
173
|
+
|
|
174
|
+
async handle(job: { id: number; payload: Record<string, any> }) {
|
|
175
|
+
const payload = job.payload as {
|
|
176
|
+
bucket?: string;
|
|
177
|
+
key?: string;
|
|
178
|
+
fileName?: string;
|
|
179
|
+
uploadItemId?: number | null;
|
|
180
|
+
userId?: number;
|
|
181
|
+
courseId?: number;
|
|
182
|
+
sessionId?: number;
|
|
183
|
+
lessonId?: number;
|
|
184
|
+
};
|
|
185
|
+
|
|
186
|
+
const bucket = String(payload.bucket ?? '').trim();
|
|
187
|
+
const key = String(payload.key ?? '').trim();
|
|
188
|
+
const fileName = String(payload.fileName ?? '').trim() || basename(key);
|
|
189
|
+
const userId = Number(payload.userId ?? 0);
|
|
190
|
+
const courseId = Number(payload.courseId ?? 0);
|
|
191
|
+
const sessionId = Number(payload.sessionId ?? 0);
|
|
192
|
+
const lessonId = Number(payload.lessonId ?? 0);
|
|
193
|
+
|
|
194
|
+
if (!bucket || !key || !userId || !courseId || !sessionId || !lessonId) {
|
|
195
|
+
throw new BadRequestException('Invalid LMS bulk upload queue payload.');
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
const storage = await this.resolveStorageContext();
|
|
199
|
+
const tempCredentials = await this.fileService.getTemporaryCredentials({
|
|
200
|
+
accessKeyId: storage.accessKeyId,
|
|
201
|
+
secretAccessKey: storage.secretAccessKey,
|
|
202
|
+
sessionToken: storage.sessionToken,
|
|
203
|
+
region: storage.region,
|
|
204
|
+
roleArn: storage.roleArn,
|
|
205
|
+
externalId: storage.externalId,
|
|
206
|
+
durationSeconds: storage.durationSeconds,
|
|
207
|
+
sessionName: `lms-bulk-upload-u${userId}-${Date.now()}`,
|
|
208
|
+
});
|
|
209
|
+
|
|
210
|
+
const workDir = await fs.mkdtemp(join(tmpdir(), 'lms-bulk-upload-'));
|
|
211
|
+
const tempFilePath = join(
|
|
212
|
+
workDir,
|
|
213
|
+
`${Date.now()}-${basename(fileName || key)}${extname(fileName || key) || '.mp4'}`,
|
|
214
|
+
);
|
|
215
|
+
|
|
216
|
+
try {
|
|
217
|
+
const s3 = new S3Client({
|
|
218
|
+
region: storage.region,
|
|
219
|
+
credentials: {
|
|
220
|
+
accessKeyId: tempCredentials.AccessKeyId,
|
|
221
|
+
secretAccessKey: tempCredentials.SecretAccessKey,
|
|
222
|
+
sessionToken: tempCredentials.SessionToken,
|
|
223
|
+
},
|
|
224
|
+
});
|
|
225
|
+
|
|
226
|
+
const response = await s3.send(
|
|
227
|
+
new GetObjectCommand({ Bucket: bucket, Key: key }),
|
|
228
|
+
);
|
|
229
|
+
const body = response.Body as any;
|
|
230
|
+
if (!body) {
|
|
231
|
+
throw new BadRequestException('Could not read the S3 object body.');
|
|
232
|
+
}
|
|
233
|
+
|
|
234
|
+
const readable = typeof body.pipe === 'function' ? body : Readable.fromWeb(body);
|
|
235
|
+
await pipeline(readable, createWriteStream(tempFilePath));
|
|
236
|
+
|
|
237
|
+
const uploaded = await this.fileService.uploadFromPath(
|
|
238
|
+
'lms/lessons/originals',
|
|
239
|
+
tempFilePath,
|
|
240
|
+
{
|
|
241
|
+
originalname: fileName,
|
|
242
|
+
filename: fileName,
|
|
243
|
+
mimetype: this.inferVideoMimeType(fileName || key),
|
|
244
|
+
},
|
|
245
|
+
);
|
|
246
|
+
|
|
247
|
+
await this.courseVideoConversionService.enqueueConversion({
|
|
248
|
+
userId,
|
|
249
|
+
courseId,
|
|
250
|
+
sessionId,
|
|
251
|
+
lessonId,
|
|
252
|
+
originalFileId: uploaded.id,
|
|
253
|
+
});
|
|
254
|
+
|
|
255
|
+
return {
|
|
256
|
+
success: true,
|
|
257
|
+
queueJobId: job.id,
|
|
258
|
+
originalFileId: uploaded.id,
|
|
259
|
+
};
|
|
260
|
+
} catch (error) {
|
|
261
|
+
if (Number.isInteger(Number(payload.uploadItemId)) && Number(payload.uploadItemId) > 0) {
|
|
262
|
+
await this.prisma.$executeRawUnsafe(
|
|
263
|
+
`UPDATE "lms_bulk_upload_item"
|
|
264
|
+
SET "status" = 'error', "error_message" = $2, "progress_percent" = 100, "updated_at" = NOW()
|
|
265
|
+
WHERE "id" = $1`,
|
|
266
|
+
Number(payload.uploadItemId),
|
|
267
|
+
error instanceof Error ? error.message : 'Unknown error',
|
|
268
|
+
).catch(() => undefined);
|
|
269
|
+
}
|
|
270
|
+
|
|
271
|
+
throw error;
|
|
272
|
+
} finally {
|
|
273
|
+
await fs.rm(workDir, { recursive: true, force: true }).catch(() => undefined);
|
|
274
|
+
}
|
|
275
|
+
}
|
|
276
|
+
|
|
277
|
+
private async findUploadItem(
|
|
278
|
+
uploadId: string | undefined,
|
|
279
|
+
fileName: string,
|
|
280
|
+
key: string,
|
|
281
|
+
): Promise<{ id: number; userId: number } | null> {
|
|
282
|
+
const rows = await this.prisma.$queryRawUnsafe<Array<{ id: number; user_id: number }>>(
|
|
283
|
+
`SELECT i."id", s."user_id"
|
|
284
|
+
FROM "lms_bulk_upload_item" i
|
|
285
|
+
INNER JOIN "lms_bulk_upload_session" s ON s."id" = i."session_id"
|
|
286
|
+
WHERE ($1::text IS NOT NULL AND i."upload_id" = $1)
|
|
287
|
+
OR i."file_name" = $2
|
|
288
|
+
OR i."uploaded_key" = $3
|
|
289
|
+
OR i."uploaded_key" = $2
|
|
290
|
+
ORDER BY i."id" DESC
|
|
291
|
+
LIMIT 1`,
|
|
292
|
+
uploadId || null,
|
|
293
|
+
fileName,
|
|
294
|
+
key,
|
|
295
|
+
);
|
|
296
|
+
|
|
297
|
+
const row = rows[0];
|
|
298
|
+
if (!row) {
|
|
299
|
+
return null;
|
|
300
|
+
}
|
|
301
|
+
|
|
302
|
+
return { id: Number(row.id), userId: Number(row.user_id) };
|
|
303
|
+
}
|
|
304
|
+
|
|
305
|
+
private async findLessonMatch(fileName: string) {
|
|
306
|
+
const parsed = this.parseFilename(fileName);
|
|
307
|
+
if (!parsed) {
|
|
308
|
+
return null;
|
|
309
|
+
}
|
|
310
|
+
|
|
311
|
+
const courses = await (this.prisma as any).course.findMany({
|
|
312
|
+
where: {
|
|
313
|
+
code: { equals: parsed.courseCode, mode: 'insensitive' },
|
|
314
|
+
},
|
|
315
|
+
select: {
|
|
316
|
+
id: true,
|
|
317
|
+
code: true,
|
|
318
|
+
title: true,
|
|
319
|
+
name: true,
|
|
320
|
+
slug: true,
|
|
321
|
+
course_image: {
|
|
322
|
+
select: {
|
|
323
|
+
file_id: true,
|
|
324
|
+
image_type: { select: { slug: true } },
|
|
325
|
+
},
|
|
326
|
+
},
|
|
327
|
+
course_module: {
|
|
328
|
+
orderBy: { order: 'asc' },
|
|
329
|
+
select: {
|
|
330
|
+
id: true,
|
|
331
|
+
title: true,
|
|
332
|
+
order: true,
|
|
333
|
+
course_lesson: {
|
|
334
|
+
orderBy: { order: 'asc' },
|
|
335
|
+
select: {
|
|
336
|
+
id: true,
|
|
337
|
+
title: true,
|
|
338
|
+
order: true,
|
|
339
|
+
},
|
|
340
|
+
},
|
|
341
|
+
},
|
|
342
|
+
},
|
|
343
|
+
},
|
|
344
|
+
});
|
|
345
|
+
|
|
346
|
+
for (const course of courses ?? []) {
|
|
347
|
+
for (const session of course.course_module ?? []) {
|
|
348
|
+
const sessionCode = this.normalizeComparableText(this.formatCode('S', Number(session.order ?? 0)));
|
|
349
|
+
if (sessionCode !== parsed.sessionCode) {
|
|
350
|
+
continue;
|
|
351
|
+
}
|
|
352
|
+
|
|
353
|
+
for (const lesson of session.course_lesson ?? []) {
|
|
354
|
+
if (this.normalizeComparableText(lesson.title) !== parsed.lessonTitle) {
|
|
355
|
+
continue;
|
|
356
|
+
}
|
|
357
|
+
|
|
358
|
+
const logo = course.course_image?.find(
|
|
359
|
+
(image: any) => image.image_type?.slug === 'course-logo',
|
|
360
|
+
);
|
|
361
|
+
|
|
362
|
+
return {
|
|
363
|
+
courseId: Number(course.id),
|
|
364
|
+
courseTitle: String(course.title ?? course.name ?? ''),
|
|
365
|
+
courseSlug: String(course.slug ?? ''),
|
|
366
|
+
courseLogoFileId: logo?.file_id ? Number(logo.file_id) : null,
|
|
367
|
+
sessionId: Number(session.id),
|
|
368
|
+
sessionTitle: String(session.title ?? ''),
|
|
369
|
+
lessonId: Number(lesson.id),
|
|
370
|
+
lessonTitle: String(lesson.title ?? ''),
|
|
371
|
+
};
|
|
372
|
+
}
|
|
373
|
+
}
|
|
374
|
+
}
|
|
375
|
+
|
|
376
|
+
return null;
|
|
377
|
+
}
|
|
378
|
+
|
|
379
|
+
private parseFilename(fileName: string) {
|
|
380
|
+
const base = this.normalizeComparableText(fileName.replace(/\.[^.]+$/, ''));
|
|
381
|
+
const parts = base.split('_').filter(Boolean);
|
|
382
|
+
if (parts.length < 3) {
|
|
383
|
+
return null;
|
|
384
|
+
}
|
|
385
|
+
|
|
386
|
+
return {
|
|
387
|
+
sessionCode: this.normalizeComparableText(parts[0]),
|
|
388
|
+
courseCode: this.normalizeComparableText(parts[1]),
|
|
389
|
+
lessonTitle: this.normalizeComparableText(parts.slice(2).join('_')),
|
|
390
|
+
};
|
|
391
|
+
}
|
|
392
|
+
|
|
393
|
+
private normalizeWebhookBody(body: unknown) {
|
|
394
|
+
const raw = typeof body === 'string' ? this.safeJsonParse(body) : (body ?? {});
|
|
395
|
+
const records = Array.isArray((raw as any).Records)
|
|
396
|
+
? (raw as any).Records
|
|
397
|
+
: Array.isArray((raw as any).records)
|
|
398
|
+
? (raw as any).records
|
|
399
|
+
: [];
|
|
400
|
+
const record = records[0] ?? (raw as any).detail ?? raw;
|
|
401
|
+
const bucket = this.firstString(
|
|
402
|
+
(raw as any)?.bucket?.name,
|
|
403
|
+
(raw as any)?.bucketName,
|
|
404
|
+
record?.s3?.bucket?.name,
|
|
405
|
+
record?.bucket?.name,
|
|
406
|
+
record?.detail?.bucket?.name,
|
|
407
|
+
);
|
|
408
|
+
const key = this.decodeKey(
|
|
409
|
+
this.firstString(
|
|
410
|
+
(raw as any)?.key,
|
|
411
|
+
(raw as any)?.object?.key,
|
|
412
|
+
record?.s3?.object?.key,
|
|
413
|
+
record?.object?.key,
|
|
414
|
+
record?.detail?.object?.key,
|
|
415
|
+
),
|
|
416
|
+
);
|
|
417
|
+
|
|
418
|
+
return {
|
|
419
|
+
bucket,
|
|
420
|
+
key,
|
|
421
|
+
fileName:
|
|
422
|
+
this.firstString(
|
|
423
|
+
(raw as any)?.fileName,
|
|
424
|
+
(raw as any)?.file_name,
|
|
425
|
+
record?.fileName,
|
|
426
|
+
record?.file_name,
|
|
427
|
+
) || (key ? basename(key) : ''),
|
|
428
|
+
uploadId: this.firstString(
|
|
429
|
+
(raw as any)?.uploadId,
|
|
430
|
+
(raw as any)?.upload_id,
|
|
431
|
+
record?.uploadId,
|
|
432
|
+
record?.upload_id,
|
|
433
|
+
),
|
|
434
|
+
userId:
|
|
435
|
+
Number(
|
|
436
|
+
(raw as any)?.userId ?? (raw as any)?.user_id ?? record?.userId ?? 0,
|
|
437
|
+
) || 0,
|
|
438
|
+
};
|
|
439
|
+
}
|
|
440
|
+
|
|
441
|
+
private async resolveStorageContext(): Promise<StorageContext> {
|
|
442
|
+
const settings = await this.settingService.getSettingValues([
|
|
443
|
+
'lms-bulk-upload-storage-profile-id',
|
|
444
|
+
'lms-bulk-upload-sts-duration-seconds',
|
|
445
|
+
]);
|
|
446
|
+
|
|
447
|
+
const storageProfileId = Number(settings['lms-bulk-upload-storage-profile-id'] ?? 0);
|
|
448
|
+
|
|
449
|
+
if (!Number.isFinite(storageProfileId) || storageProfileId <= 0) {
|
|
450
|
+
throw new BadRequestException('O perfil de armazenamento para upload em massa não está configurado. Acesse as configurações e defina o perfil de integração S3.');
|
|
451
|
+
}
|
|
452
|
+
|
|
453
|
+
const profile = await (this.prisma as any).integration_profile.findUnique({
|
|
454
|
+
where: { id: storageProfileId },
|
|
455
|
+
include: {
|
|
456
|
+
integration_type: { select: { slug: true } },
|
|
457
|
+
integration_provider: { select: { slug: true } },
|
|
458
|
+
},
|
|
459
|
+
});
|
|
460
|
+
|
|
461
|
+
if (!profile) {
|
|
462
|
+
throw new BadRequestException('Bulk upload storage profile not found.');
|
|
463
|
+
}
|
|
464
|
+
|
|
465
|
+
const config = profile.config as Record<string, any>;
|
|
466
|
+
const bucket = String(config.bucket ?? '').trim();
|
|
467
|
+
const accessKeyId = String(config.access_key_id ?? '').trim();
|
|
468
|
+
const secretAccessKey = String(config.secret_access_key ?? '').trim();
|
|
469
|
+
|
|
470
|
+
if (!bucket || !accessKeyId || !secretAccessKey) {
|
|
471
|
+
throw new BadRequestException('Bulk upload storage profile does not expose AWS credentials.');
|
|
472
|
+
}
|
|
473
|
+
|
|
474
|
+
return {
|
|
475
|
+
region: String(config.region ?? '').trim() || 'us-east-1',
|
|
476
|
+
bucket,
|
|
477
|
+
accessKeyId,
|
|
478
|
+
secretAccessKey,
|
|
479
|
+
sessionToken: String(config.session_token ?? '').trim() || undefined,
|
|
480
|
+
roleArn: String(config.role_arn ?? '').trim() || undefined,
|
|
481
|
+
externalId: String(config.external_id ?? '').trim() || undefined,
|
|
482
|
+
durationSeconds: Math.max(
|
|
483
|
+
900,
|
|
484
|
+
Number(settings['lms-bulk-upload-sts-duration-seconds'] ?? 3600) || 3600,
|
|
485
|
+
),
|
|
486
|
+
};
|
|
487
|
+
}
|
|
488
|
+
|
|
489
|
+
private inferVideoMimeType(filename: string) {
|
|
490
|
+
switch (extname(filename).toLowerCase()) {
|
|
491
|
+
case '.mp4':
|
|
492
|
+
case '.m4v':
|
|
493
|
+
return 'video/mp4';
|
|
494
|
+
case '.mov':
|
|
495
|
+
return 'video/quicktime';
|
|
496
|
+
case '.mkv':
|
|
497
|
+
return 'video/x-matroska';
|
|
498
|
+
case '.webm':
|
|
499
|
+
return 'video/webm';
|
|
500
|
+
case '.avi':
|
|
501
|
+
return 'video/x-msvideo';
|
|
502
|
+
case '.wmv':
|
|
503
|
+
return 'video/x-ms-wmv';
|
|
504
|
+
case '.flv':
|
|
505
|
+
return 'video/x-flv';
|
|
506
|
+
case '.ogv':
|
|
507
|
+
return 'video/ogg';
|
|
508
|
+
case '.m2ts':
|
|
509
|
+
case '.ts':
|
|
510
|
+
return 'video/mp2t';
|
|
511
|
+
default:
|
|
512
|
+
return 'video/mp4';
|
|
513
|
+
}
|
|
514
|
+
}
|
|
515
|
+
|
|
516
|
+
private decodeKey(value: string) {
|
|
517
|
+
if (!value) {
|
|
518
|
+
return value;
|
|
519
|
+
}
|
|
520
|
+
|
|
521
|
+
try {
|
|
522
|
+
return decodeURIComponent(value.replace(/\+/g, ' '));
|
|
523
|
+
} catch {
|
|
524
|
+
return value;
|
|
525
|
+
}
|
|
526
|
+
}
|
|
527
|
+
|
|
528
|
+
private firstString(...values: Array<unknown>) {
|
|
529
|
+
for (const value of values) {
|
|
530
|
+
const normalized = String(value ?? '').trim();
|
|
531
|
+
if (normalized) {
|
|
532
|
+
return normalized;
|
|
533
|
+
}
|
|
534
|
+
}
|
|
535
|
+
|
|
536
|
+
return '';
|
|
537
|
+
}
|
|
538
|
+
|
|
539
|
+
private normalizeComparableText(value: string) {
|
|
540
|
+
return String(value ?? '')
|
|
541
|
+
.normalize('NFD')
|
|
542
|
+
.replace(/[\u0300-\u036f]/g, '')
|
|
543
|
+
.replace(/[^a-zA-Z0-9]+/g, '_')
|
|
544
|
+
.replace(/_+/g, '_')
|
|
545
|
+
.replace(/^_+|_+$/g, '')
|
|
546
|
+
.toLowerCase();
|
|
547
|
+
}
|
|
548
|
+
|
|
549
|
+
private formatCode(prefix: 'S' | 'A', order: number) {
|
|
550
|
+
return `${prefix}${String(Math.max(order, 0)).padStart(2, '0')}`;
|
|
551
|
+
}
|
|
552
|
+
|
|
553
|
+
private safeJsonParse(value: string) {
|
|
554
|
+
try {
|
|
555
|
+
return JSON.parse(value);
|
|
556
|
+
} catch {
|
|
557
|
+
return {};
|
|
558
|
+
}
|
|
559
|
+
}
|
|
560
|
+
}
|