feat: add recommended courses and quiz multiple attempts

- Add is_recommended field to Course model
- Add allow_multiple_attempts field to Quiz model
- Create RecommendedCoursesController for admin management
  - List all approved courses
  - Get course by ID
  - Toggle recommendation status
- Add is_recommended filter to CoursesService.ListCourses
- Add allow_multiple_attempts to quiz update and response types
- Update ChaptersLessonService.updateQuiz to support allow_multiple_attempts
This commit is contained in:
JakkrapartXD 2026-02-11 13:49:43 +07:00
parent 623f797763
commit f7330a7b27
17 changed files with 3963 additions and 5 deletions

View file

@ -0,0 +1,75 @@
import { Body, Get, Path, Put, Request, Response, Route, Security, SuccessResponse, Tags } from 'tsoa';
import { ValidationError } from '../middleware/errorHandler';
import { RecommendedCoursesService } from '../services/RecommendedCourses.service';
import {
ListApprovedCoursesResponse,
GetCourseByIdResponse,
ToggleRecommendedRequest,
ToggleRecommendedResponse
} from '../types/RecommendedCourses.types';
@Route('api/admin/recommended-courses')
@Tags('Admin/RecommendedCourses')
export class RecommendedCoursesController {
/**
* ()
* List all approved courses (for managing recommendations)
*/
@Get()
@Security('jwt', ['admin'])
@SuccessResponse('200', 'Approved courses retrieved successfully')
@Response('401', 'Unauthorized')
@Response('403', 'Forbidden - Admin only')
public async listApprovedCourses(@Request() request: any): Promise<ListApprovedCoursesResponse> {
const token = request.headers.authorization?.replace('Bearer ', '');
if (!token) {
throw new ValidationError('No token provided');
}
return await RecommendedCoursesService.listApprovedCourses();
}
/**
* ID
* Get course by ID
* @param courseId - / Course ID
*/
@Get('{courseId}')
@Security('jwt', ['admin'])
@SuccessResponse('200', 'Course retrieved successfully')
@Response('400', 'Course is not approved')
@Response('401', 'Unauthorized')
@Response('403', 'Forbidden - Admin only')
@Response('404', 'Course not found')
public async getCourseById(@Request() request: any, @Path() courseId: number): Promise<GetCourseByIdResponse> {
const token = request.headers.authorization?.replace('Bearer ', '');
if (!token) {
throw new ValidationError('No token provided');
}
return await RecommendedCoursesService.getCourseById(courseId);
}
/**
*
* Toggle course recommendation status
* @param courseId - / Course ID
*/
@Put('{courseId}/toggle')
@Security('jwt', ['admin'])
@SuccessResponse('200', 'Recommendation status updated successfully')
@Response('400', 'Only approved courses can be recommended')
@Response('401', 'Unauthorized')
@Response('403', 'Forbidden - Admin only')
@Response('404', 'Course not found')
public async toggleRecommended(
@Request() request: any,
@Path() courseId: number,
@Body() body: ToggleRecommendedRequest
): Promise<ToggleRecommendedResponse> {
const token = request.headers.authorization?.replace('Bearer ', '');
if (!token) {
throw new ValidationError('No token provided');
}
return await RecommendedCoursesService.toggleRecommended(token, courseId, body.is_recommended);
}
}