package handler import ( "apskel-pos-be/internal/appcontext" "apskel-pos-be/internal/constants" "apskel-pos-be/internal/contract" "apskel-pos-be/internal/logger" "apskel-pos-be/internal/models" "apskel-pos-be/internal/util" "strconv" "github.com/gin-gonic/gin" "github.com/google/uuid" ) type IngredientHandler struct { ingredientService IngredientService } func NewIngredientHandler(ingredientService IngredientService) *IngredientHandler { return &IngredientHandler{ ingredientService: ingredientService, } } func (h *IngredientHandler) Create(c *gin.Context) { ctx := c.Request.Context() contextInfo := appcontext.FromGinContext(ctx) var request models.CreateIngredientRequest if err := c.ShouldBindJSON(&request); err != nil { logger.FromContext(c.Request.Context()).WithError(err).Error("IngredientHandler::Create -> request binding failed") validationResponseError := contract.NewResponseError(constants.MissingFieldErrorCode, constants.RequestEntity, err.Error()) util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "IngredientHandler::Create") return } request.OrganizationID = contextInfo.OrganizationID ingredientResponse, err := h.ingredientService.CreateIngredient(ctx, &request) if err != nil { logger.FromContext(ctx).WithError(err).Error("IngredientHandler::Create -> Failed to create ingredient from service") validationResponseError := contract.NewResponseError(constants.InternalServerErrorCode, constants.RequestEntity, err.Error()) util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "IngredientHandler::Create") return } util.HandleResponse(c.Writer, c.Request, contract.BuildSuccessResponse(ingredientResponse), "IngredientHandler::Create") } func (h *IngredientHandler) GetByID(c *gin.Context) { ctx := c.Request.Context() idStr := c.Param("id") id, err := uuid.Parse(idStr) if err != nil { logger.FromContext(ctx).WithError(err).Error("IngredientHandler::GetByID -> Invalid ingredient ID") validationResponseError := contract.NewResponseError(constants.MalformedFieldErrorCode, constants.RequestEntity, "Invalid ingredient ID") util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "IngredientHandler::GetByID") return } ingredientResponse, err := h.ingredientService.GetIngredientByID(ctx, id) if err != nil { logger.FromContext(ctx).WithError(err).Error("IngredientHandler::GetByID -> Failed to get ingredient from service") validationResponseError := contract.NewResponseError(constants.NotFoundErrorCode, constants.RequestEntity, "Ingredient not found") util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "IngredientHandler::GetByID") return } util.HandleResponse(c.Writer, c.Request, contract.BuildSuccessResponse(ingredientResponse), "IngredientHandler::GetByID") } func (h *IngredientHandler) GetAll(c *gin.Context) { ctx := c.Request.Context() contextInfo := appcontext.FromGinContext(ctx) // Get query parameters pageStr := c.DefaultQuery("page", "1") limitStr := c.DefaultQuery("limit", "10") search := c.Query("search") outletIDStr := c.Query("outlet_id") page, err := strconv.Atoi(pageStr) if err != nil { logger.FromContext(ctx).WithError(err).Error("IngredientHandler::GetAll -> Invalid page parameter") validationResponseError := contract.NewResponseError(constants.MalformedFieldErrorCode, constants.RequestEntity, "Invalid page parameter") util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "IngredientHandler::GetAll") return } limit, err := strconv.Atoi(limitStr) if err != nil { logger.FromContext(ctx).WithError(err).Error("IngredientHandler::GetAll -> Invalid limit parameter") validationResponseError := contract.NewResponseError(constants.MalformedFieldErrorCode, constants.RequestEntity, "Invalid limit parameter") util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "IngredientHandler::GetAll") return } var outletID *uuid.UUID if outletIDStr != "" { parsedOutletID, err := uuid.Parse(outletIDStr) if err != nil { logger.FromContext(ctx).WithError(err).Error("IngredientHandler::GetAll -> Invalid outlet ID") validationResponseError := contract.NewResponseError(constants.MalformedFieldErrorCode, constants.RequestEntity, "Invalid outlet ID") util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "IngredientHandler::GetAll") return } outletID = &parsedOutletID } ingredientResponse, err := h.ingredientService.ListIngredients(ctx, contextInfo.OrganizationID, outletID, page, limit, search) if err != nil { logger.FromContext(ctx).WithError(err).Error("IngredientHandler::GetAll -> Failed to get ingredients from service") validationResponseError := contract.NewResponseError(constants.InternalServerErrorCode, constants.RequestEntity, "Failed to get ingredients") util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "IngredientHandler::GetAll") return } util.HandleResponse(c.Writer, c.Request, contract.BuildSuccessResponse(ingredientResponse), "IngredientHandler::GetAll") } func (h *IngredientHandler) Update(c *gin.Context) { ctx := c.Request.Context() idStr := c.Param("id") id, err := uuid.Parse(idStr) if err != nil { logger.FromContext(ctx).WithError(err).Error("IngredientHandler::Update -> Invalid ingredient ID") validationResponseError := contract.NewResponseError(constants.MalformedFieldErrorCode, constants.RequestEntity, "Invalid ingredient ID") util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "IngredientHandler::Update") return } var request models.UpdateIngredientRequest if err := c.ShouldBindJSON(&request); err != nil { logger.FromContext(ctx).WithError(err).Error("IngredientHandler::Update -> request binding failed") validationResponseError := contract.NewResponseError(constants.MissingFieldErrorCode, constants.RequestEntity, "Invalid request body") util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "IngredientHandler::Update") return } ingredientResponse, err := h.ingredientService.UpdateIngredient(ctx, id, &request) if err != nil { logger.FromContext(ctx).WithError(err).Error("IngredientHandler::Update -> Failed to update ingredient from service") validationResponseError := contract.NewResponseError(constants.InternalServerErrorCode, constants.RequestEntity, err.Error()) util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "IngredientHandler::Update") return } util.HandleResponse(c.Writer, c.Request, contract.BuildSuccessResponse(ingredientResponse), "IngredientHandler::Update") } func (h *IngredientHandler) Delete(c *gin.Context) { ctx := c.Request.Context() idStr := c.Param("id") id, err := uuid.Parse(idStr) if err != nil { logger.FromContext(ctx).WithError(err).Error("IngredientHandler::Delete -> Invalid ingredient ID") validationResponseError := contract.NewResponseError(constants.MalformedFieldErrorCode, constants.RequestEntity, "Invalid ingredient ID") util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "IngredientHandler::Delete") return } err = h.ingredientService.DeleteIngredient(ctx, id) if err != nil { logger.FromContext(ctx).WithError(err).Error("IngredientHandler::Delete -> Failed to delete ingredient from service") validationResponseError := contract.NewResponseError(constants.InternalServerErrorCode, constants.RequestEntity, err.Error()) util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "IngredientHandler::Delete") return } util.HandleResponse(c.Writer, c.Request, contract.BuildSuccessResponse(map[string]interface{}{ "message": "Ingredient deleted successfully", }), "IngredientHandler::Delete") }