package handlers import ( "Yimaru-Backend/internal/domain" "Yimaru-Backend/internal/services/programs" "context" "errors" "strconv" "github.com/gofiber/fiber/v2" ) // CreateProgram godoc // @Summary Create program // @Description Create a top-level LMS program. Optional sort_order inserts at that global ordering; omit it to append after the current highest sort_order. Unique constraint applies to sort_order. // @Tags programs // @Accept json // @Produce json // @Param body body domain.CreateProgramInput true "Program" // @Success 201 {object} domain.Response // @Failure 400 {object} domain.ErrorResponse // @Failure 500 {object} domain.ErrorResponse // @Router /api/v1/programs [post] func (h *Handler) CreateProgram(c *fiber.Ctx) error { var req domain.CreateProgramInput if err := c.BodyParser(&req); err != nil { return c.Status(fiber.StatusBadRequest).JSON(domain.ErrorResponse{ Message: "Invalid request body", Error: err.Error(), }) } if valErrs, ok := h.validator.Validate(c, req); !ok { return c.Status(fiber.StatusBadRequest).JSON(domain.ErrorResponse{ Message: "Validation failed", Error: firstValidationError(valErrs), }) } p, err := h.programSvc.Create(c.Context(), req) if err != nil { return c.Status(fiber.StatusInternalServerError).JSON(domain.ErrorResponse{ Message: "Failed to create program", Error: err.Error(), }) } actorID := c.Locals("user_id").(int64) actorRole := string(c.Locals("role").(domain.Role)) ip := c.IP() ua := c.Get("User-Agent") go h.activityLogSvc.RecordAction(context.Background(), &actorID, &actorRole, domain.ActionProgramCreated, domain.ResourceProgram, &p.ID, "Created program: "+p.Name, nil, &ip, &ua) return c.Status(fiber.StatusCreated).JSON(domain.Response{ Message: "Program created successfully", Data: p, Success: true, StatusCode: fiber.StatusCreated, }) } // ListPrograms godoc // @Summary List programs // @Description Paginated list of programs // @Tags programs // @Produce json // @Param limit query int false "Page size" default(20) // @Param offset query int false "Offset" default(0) // @Success 200 {object} domain.Response // @Failure 500 {object} domain.ErrorResponse // @Router /api/v1/programs [get] func (h *Handler) ListPrograms(c *fiber.Ctx) error { limit, _ := strconv.Atoi(c.Query("limit", "20")) offset, _ := strconv.Atoi(c.Query("offset", "0")) items, total, err := h.programSvc.List(c.Context(), int32(limit), int32(offset)) if err != nil { return c.Status(fiber.StatusInternalServerError).JSON(domain.ErrorResponse{ Message: "Failed to list programs", Error: err.Error(), }) } uid := c.Locals("user_id").(int64) role := c.Locals("role").(domain.Role) for i := range items { if err := h.lmsProgressSvc.ApplyAccessProgram(c.Context(), role, uid, &items[i]); err != nil { return c.Status(fiber.StatusInternalServerError).JSON(domain.ErrorResponse{ Message: "Failed to build program list", Error: err.Error(), }) } } return c.JSON(domain.Response{ Message: "Programs retrieved successfully", Data: fiber.Map{ "programs": items, "total_count": total, "limit": limit, "offset": offset, }, Success: true, StatusCode: fiber.StatusOK, }) } // GetProgram godoc // @Summary Get program by ID // @Tags programs // @Produce json // @Param id path int true "Program ID" // @Success 200 {object} domain.Response // @Failure 404 {object} domain.ErrorResponse // @Router /api/v1/programs/{id} [get] func (h *Handler) GetProgram(c *fiber.Ctx) error { id, err := strconv.ParseInt(c.Params("id"), 10, 64) if err != nil { return c.Status(fiber.StatusBadRequest).JSON(domain.ErrorResponse{ Message: "Invalid program id", Error: err.Error(), }) } p, err := h.programSvc.GetByID(c.Context(), id) if err != nil { if errors.Is(err, programs.ErrProgramNotFound) { return c.Status(fiber.StatusNotFound).JSON(domain.ErrorResponse{ Message: "Program not found", Error: err.Error(), }) } return c.Status(fiber.StatusInternalServerError).JSON(domain.ErrorResponse{ Message: "Failed to load program", Error: err.Error(), }) } uid := c.Locals("user_id").(int64) role := c.Locals("role").(domain.Role) if err := h.lmsProgressSvc.ApplyAccessProgram(c.Context(), role, uid, &p); err != nil { return c.Status(fiber.StatusInternalServerError).JSON(domain.ErrorResponse{ Message: "Failed to evaluate program access", Error: err.Error(), }) } if err := lmsBlockIfInaccessible(c, p.Access); err != nil { return err } return c.JSON(domain.Response{ Message: "Program retrieved successfully", Data: p, Success: true, StatusCode: fiber.StatusOK, }) } // UpdateProgram godoc // @Summary Update program // @Tags programs // @Accept json // @Produce json // @Param id path int true "Program ID" // @Param body body domain.UpdateProgramInput true "Fields to update" // @Success 200 {object} domain.Response // @Failure 400 {object} domain.ErrorResponse // @Failure 404 {object} domain.ErrorResponse // @Router /api/v1/programs/{id} [put] func (h *Handler) UpdateProgram(c *fiber.Ctx) error { id, err := strconv.ParseInt(c.Params("id"), 10, 64) if err != nil { return c.Status(fiber.StatusBadRequest).JSON(domain.ErrorResponse{ Message: "Invalid program id", Error: err.Error(), }) } var req domain.UpdateProgramInput if err := c.BodyParser(&req); err != nil { return c.Status(fiber.StatusBadRequest).JSON(domain.ErrorResponse{ Message: "Invalid request body", Error: err.Error(), }) } if valErrs, ok := h.validator.Validate(c, req); !ok { return c.Status(fiber.StatusBadRequest).JSON(domain.ErrorResponse{ Message: "Validation failed", Error: firstValidationError(valErrs), }) } p, err := h.programSvc.Update(c.Context(), id, req) if err != nil { if errors.Is(err, programs.ErrProgramNotFound) { return c.Status(fiber.StatusNotFound).JSON(domain.ErrorResponse{ Message: "Program not found", Error: err.Error(), }) } return c.Status(fiber.StatusInternalServerError).JSON(domain.ErrorResponse{ Message: "Failed to update program", Error: err.Error(), }) } actorID := c.Locals("user_id").(int64) actorRole := string(c.Locals("role").(domain.Role)) ip := c.IP() ua := c.Get("User-Agent") go h.activityLogSvc.RecordAction(context.Background(), &actorID, &actorRole, domain.ActionProgramUpdated, domain.ResourceProgram, &p.ID, "Updated program: "+p.Name, nil, &ip, &ua) return c.JSON(domain.Response{ Message: "Program updated successfully", Data: p, Success: true, StatusCode: fiber.StatusOK, }) } // DeleteProgram godoc // @Summary Delete program // @Tags programs // @Param id path int true "Program ID" // @Success 200 {object} domain.Response // @Failure 404 {object} domain.ErrorResponse // @Router /api/v1/programs/{id} [delete] func (h *Handler) DeleteProgram(c *fiber.Ctx) error { id, err := strconv.ParseInt(c.Params("id"), 10, 64) if err != nil { return c.Status(fiber.StatusBadRequest).JSON(domain.ErrorResponse{ Message: "Invalid program id", Error: err.Error(), }) } if err := h.programSvc.Delete(c.Context(), id); err != nil { if errors.Is(err, programs.ErrProgramNotFound) { return c.Status(fiber.StatusNotFound).JSON(domain.ErrorResponse{ Message: "Program not found", Error: err.Error(), }) } return c.Status(fiber.StatusInternalServerError).JSON(domain.ErrorResponse{ Message: "Failed to delete program", Error: err.Error(), }) } actorID := c.Locals("user_id").(int64) actorRole := string(c.Locals("role").(domain.Role)) ip := c.IP() ua := c.Get("User-Agent") go h.activityLogSvc.RecordAction(context.Background(), &actorID, &actorRole, domain.ActionProgramDeleted, domain.ResourceProgram, &id, "Deleted program", nil, &ip, &ua) return c.JSON(domain.Response{ Message: "Program deleted successfully", Success: true, StatusCode: fiber.StatusOK, }) } func firstValidationError(errs map[string]string) string { for _, v := range errs { return v } return "validation error" }