436 lines
12 KiB
Go
436 lines
12 KiB
Go
package controller
|
|
|
|
import (
|
|
"encoding/json"
|
|
"jaecoo-be/app/module/products/request"
|
|
"jaecoo-be/app/module/products/service"
|
|
"jaecoo-be/utils/paginator"
|
|
"strconv"
|
|
|
|
"github.com/gofiber/fiber/v2"
|
|
|
|
utilRes "jaecoo-be/utils/response"
|
|
)
|
|
|
|
type productsController struct {
|
|
productsService service.ProductsService
|
|
}
|
|
|
|
type ProductsController interface {
|
|
All(c *fiber.Ctx) error
|
|
Show(c *fiber.Ctx) error
|
|
Save(c *fiber.Ctx) error
|
|
Update(c *fiber.Ctx) error
|
|
Delete(c *fiber.Ctx) error
|
|
Approve(c *fiber.Ctx) error
|
|
Reject(c *fiber.Ctx) error
|
|
Comment(c *fiber.Ctx) error
|
|
Viewer(c *fiber.Ctx) error
|
|
}
|
|
|
|
func NewProductsController(productsService service.ProductsService) ProductsController {
|
|
return &productsController{
|
|
productsService: productsService,
|
|
}
|
|
}
|
|
|
|
// All Products
|
|
// @Summary Get all Products
|
|
// @Description API for getting all Products
|
|
// @Tags Products
|
|
// @Security Bearer
|
|
// @Param X-Client-Key header string true "Insert the X-Client-Key"
|
|
// @Param req query request.ProductsQueryRequestContext false "query parameters"
|
|
// @Param req query paginator.Pagination false "pagination parameters"
|
|
// @Success 200 {object} response.Response
|
|
// @Failure 400 {object} response.BadRequestError
|
|
// @Failure 401 {object} response.UnauthorizedError
|
|
// @Failure 500 {object} response.InternalServerError
|
|
// @Router /products [get]
|
|
func (_i *productsController) All(c *fiber.Ctx) error {
|
|
paginate, err := paginator.Paginate(c)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
reqContext := request.ProductsQueryRequestContext{
|
|
Title: c.Query("title"),
|
|
Variant: c.Query("variant"),
|
|
}
|
|
req := reqContext.ToParamRequest()
|
|
req.Pagination = paginate
|
|
|
|
productsData, paging, err := _i.productsService.GetAll(req)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return utilRes.Resp(c, utilRes.Response{
|
|
Success: true,
|
|
Messages: utilRes.Messages{"Products list successfully retrieved"},
|
|
Data: productsData,
|
|
Meta: paging,
|
|
})
|
|
}
|
|
|
|
// Show Product
|
|
// @Summary Get Product by ID
|
|
// @Description API for getting Product by ID
|
|
// @Tags Products
|
|
// @Security Bearer
|
|
// @Param X-Client-Key header string true "Insert the X-Client-Key"
|
|
// @Param id path int true "Product ID"
|
|
// @Success 200 {object} response.Response
|
|
// @Failure 400 {object} response.BadRequestError
|
|
// @Failure 401 {object} response.UnauthorizedError
|
|
// @Failure 500 {object} response.InternalServerError
|
|
// @Router /products/{id} [get]
|
|
func (_i *productsController) Show(c *fiber.Ctx) error {
|
|
id, err := strconv.ParseUint(c.Params("id"), 10, 0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
productData, err := _i.productsService.GetOne(uint(id))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return utilRes.Resp(c, utilRes.Response{
|
|
Success: true,
|
|
Messages: utilRes.Messages{"Product successfully retrieved"},
|
|
Data: productData,
|
|
})
|
|
}
|
|
|
|
// Save Product
|
|
// @Summary Create Product
|
|
// @Description API for creating Product with file upload
|
|
// @Tags Products
|
|
// @Security Bearer
|
|
// @Param X-Client-Key header string true "Insert the X-Client-Key"
|
|
// @Param file formData file false "Upload file"
|
|
// @Param title formData string true "Product title"
|
|
// @Param variant formData string false "Product variant"
|
|
// @Param price formData string false "Product price"
|
|
// @Param colors formData string false "Product colors (JSON array)"
|
|
// @Success 200 {object} response.Response
|
|
// @Failure 400 {object} response.BadRequestError
|
|
// @Failure 401 {object} response.UnauthorizedError
|
|
// @Failure 500 {object} response.InternalServerError
|
|
// @Router /products [post]
|
|
func (_i *productsController) Save(c *fiber.Ctx) error {
|
|
// Parse multipart form
|
|
form, err := c.MultipartForm()
|
|
if err != nil {
|
|
return utilRes.Resp(c, utilRes.Response{
|
|
Success: false,
|
|
Messages: utilRes.Messages{"Failed to parse form data"},
|
|
})
|
|
}
|
|
|
|
// Extract form values
|
|
req := request.ProductsCreateRequest{
|
|
Title: c.FormValue("title"),
|
|
}
|
|
|
|
if variant := c.FormValue("variant"); variant != "" {
|
|
req.Variant = &variant
|
|
}
|
|
|
|
if price := c.FormValue("price"); price != "" {
|
|
req.Price = &price
|
|
}
|
|
|
|
// Handle colors (JSON array string)
|
|
var colors []request.ProductColorRequest
|
|
|
|
if colorsStr := c.FormValue("colors"); colorsStr != "" {
|
|
if err := json.Unmarshal([]byte(colorsStr), &colors); err != nil {
|
|
return utilRes.Resp(c, utilRes.Response{
|
|
Success: false,
|
|
Messages: utilRes.Messages{"Invalid colors format"},
|
|
})
|
|
}
|
|
req.Colors = colors
|
|
}
|
|
|
|
|
|
// Validate required fields
|
|
if req.Title == "" {
|
|
return utilRes.Resp(c, utilRes.Response{
|
|
Success: false,
|
|
Messages: utilRes.Messages{"Title is required"},
|
|
})
|
|
}
|
|
|
|
// Check if file is uploaded
|
|
if len(form.File["file"]) > 0 {
|
|
// File will be handled in service
|
|
}
|
|
|
|
dataResult, err := _i.productsService.Create(c, req)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return utilRes.Resp(c, utilRes.Response{
|
|
Success: true,
|
|
Messages: utilRes.Messages{"Product successfully created"},
|
|
Data: dataResult,
|
|
})
|
|
}
|
|
|
|
// Update Product
|
|
// @Summary Update Product
|
|
// @Description API for updating Product with file upload
|
|
// @Tags Products
|
|
// @Security Bearer
|
|
// @Param X-Client-Key header string true "Insert the X-Client-Key"
|
|
// @Param id path int true "Product ID"
|
|
// @Param file formData file false "Upload file"
|
|
// @Param title formData string false "Product title"
|
|
// @Param variant formData string false "Product variant"
|
|
// @Param price formData string false "Product price"
|
|
// @Param colors formData string false "Product colors (JSON array)"
|
|
// @Param is_active formData bool false "Product is_active"
|
|
// @Success 200 {object} response.Response
|
|
// @Failure 400 {object} response.BadRequestError
|
|
// @Failure 401 {object} response.UnauthorizedError
|
|
// @Failure 500 {object} response.InternalServerError
|
|
// @Router /products/{id} [put]
|
|
func (_i *productsController) Update(c *fiber.Ctx) error {
|
|
id, err := strconv.ParseUint(c.Params("id"), 10, 0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Parse multipart form
|
|
form, err := c.MultipartForm()
|
|
if err != nil {
|
|
return utilRes.Resp(c, utilRes.Response{
|
|
Success: false,
|
|
Messages: utilRes.Messages{"Failed to parse form data"},
|
|
})
|
|
}
|
|
|
|
// Extract form values
|
|
req := request.ProductsUpdateRequest{}
|
|
|
|
if title := c.FormValue("title"); title != "" {
|
|
req.Title = &title
|
|
}
|
|
|
|
if variant := c.FormValue("variant"); variant != "" {
|
|
req.Variant = &variant
|
|
}
|
|
|
|
if price := c.FormValue("price"); price != "" {
|
|
req.Price = &price
|
|
}
|
|
|
|
// Handle colors (JSON array string)
|
|
if colorsStr := c.FormValue("colors"); colorsStr != "" {
|
|
var colors []string
|
|
if err := json.Unmarshal([]byte(colorsStr), &colors); err == nil {
|
|
req.Colors = colors
|
|
}
|
|
}
|
|
|
|
if isActiveStr := c.FormValue("is_active"); isActiveStr != "" {
|
|
isActive := isActiveStr == "true" || isActiveStr == "1"
|
|
req.IsActive = &isActive
|
|
}
|
|
|
|
// Check if file is uploaded
|
|
if len(form.File["file"]) > 0 {
|
|
// File will be handled in service
|
|
}
|
|
|
|
dataResult, err := _i.productsService.Update(c, uint(id), req)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return utilRes.Resp(c, utilRes.Response{
|
|
Success: true,
|
|
Messages: utilRes.Messages{"Product successfully updated"},
|
|
Data: dataResult,
|
|
})
|
|
}
|
|
|
|
// Delete Product
|
|
// @Summary Delete Product
|
|
// @Description API for deleting Product (soft delete)
|
|
// @Tags Products
|
|
// @Security Bearer
|
|
// @Param X-Client-Key header string true "Insert the X-Client-Key"
|
|
// @Param id path int true "Product ID"
|
|
// @Success 200 {object} response.Response
|
|
// @Failure 400 {object} response.BadRequestError
|
|
// @Failure 401 {object} response.UnauthorizedError
|
|
// @Failure 500 {object} response.InternalServerError
|
|
// @Router /products/{id} [delete]
|
|
func (_i *productsController) Delete(c *fiber.Ctx) error {
|
|
id, err := strconv.ParseUint(c.Params("id"), 10, 0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = _i.productsService.Delete(uint(id))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return utilRes.Resp(c, utilRes.Response{
|
|
Success: true,
|
|
Messages: utilRes.Messages{"Product successfully deleted"},
|
|
})
|
|
}
|
|
|
|
// Approve Product
|
|
// @Summary Approve Product
|
|
// @Description API for approving Product (only for admin with roleId = 1)
|
|
// @Tags Products
|
|
// @Security Bearer
|
|
// @Param X-Client-Key header string true "Insert the X-Client-Key"
|
|
// @Param id path int true "Product ID"
|
|
// @Success 200 {object} response.Response
|
|
// @Failure 400 {object} response.BadRequestError
|
|
// @Failure 401 {object} response.UnauthorizedError
|
|
// @Failure 500 {object} response.InternalServerError
|
|
// @Router /products/{id}/approve [put]
|
|
func (_i *productsController) Approve(c *fiber.Ctx) error {
|
|
id, err := strconv.ParseUint(c.Params("id"), 10, 0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Get token from Authorization header
|
|
authToken := c.Get("Authorization")
|
|
if authToken == "" {
|
|
return utilRes.Resp(c, utilRes.Response{
|
|
Success: false,
|
|
Messages: utilRes.Messages{"Unauthorized: token not found"},
|
|
})
|
|
}
|
|
|
|
productData, err := _i.productsService.Approve(uint(id), authToken)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return utilRes.Resp(c, utilRes.Response{
|
|
Success: true,
|
|
Messages: utilRes.Messages{"Product successfully approved"},
|
|
Data: productData,
|
|
})
|
|
}
|
|
|
|
// Reject Product
|
|
// @Summary Reject Product
|
|
// @Description API for rejecting Product (only for admin with roleId = 1)
|
|
// @Tags Products
|
|
// @Security Bearer
|
|
// @Param X-Client-Key header string true "Insert the X-Client-Key"
|
|
// @Param id path int true "Product ID"
|
|
// @Param message body string false "Rejection message"
|
|
// @Success 200 {object} response.Response
|
|
// @Failure 400 {object} response.BadRequestError
|
|
// @Failure 401 {object} response.UnauthorizedError
|
|
// @Failure 500 {object} response.InternalServerError
|
|
// @Router /products/{id}/reject [put]
|
|
func (_i *productsController) Reject(c *fiber.Ctx) error {
|
|
id, err := strconv.ParseUint(c.Params("id"), 10, 0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Get token from Authorization header
|
|
authToken := c.Get("Authorization")
|
|
if authToken == "" {
|
|
return utilRes.Resp(c, utilRes.Response{
|
|
Success: false,
|
|
Messages: utilRes.Messages{"Unauthorized: token not found"},
|
|
})
|
|
}
|
|
|
|
// Get optional message from request body
|
|
var body struct {
|
|
Message *string `json:"message"`
|
|
}
|
|
if err := c.BodyParser(&body); err != nil {
|
|
body.Message = nil
|
|
}
|
|
|
|
productData, err := _i.productsService.Reject(uint(id), authToken, body.Message)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return utilRes.Resp(c, utilRes.Response{
|
|
Success: true,
|
|
Messages: utilRes.Messages{"Product successfully rejected"},
|
|
Data: productData,
|
|
})
|
|
}
|
|
|
|
// Comment Banner
|
|
// @Summary Comment Banner
|
|
// @Description API for comment Banner (only admin)
|
|
// @Tags Banners
|
|
// @Security BearerAuth
|
|
// @Param id path int true "Banner ID"
|
|
// @Param body body request.CommentRequest true "Comment payload"
|
|
// @Success 200 {object} response.Response
|
|
// @Router /products/{id}/comment [put]
|
|
func (_i *productsController) Comment(c *fiber.Ctx) error {
|
|
id, err := strconv.ParseUint(c.Params("id"), 10, 0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
authToken := c.Get("Authorization")
|
|
if authToken == "" {
|
|
return utilRes.Resp(c, utilRes.Response{
|
|
Success: false,
|
|
Messages: utilRes.Messages{"Unauthorized"},
|
|
})
|
|
}
|
|
|
|
var req request.CommentRequest
|
|
if err := c.BodyParser(&req); err != nil {
|
|
return utilRes.Resp(c, utilRes.Response{
|
|
Success: false,
|
|
Messages: utilRes.Messages{"Invalid request"},
|
|
})
|
|
}
|
|
|
|
bannerData, err := _i.productsService.Comment(uint(id), authToken, &req.Message)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return utilRes.Resp(c, utilRes.Response{
|
|
Success: true,
|
|
Messages: utilRes.Messages{"Komentar berhasil disimpan"},
|
|
Data: bannerData,
|
|
})
|
|
}
|
|
|
|
// Viewer Product
|
|
// @Summary Viewer Product
|
|
// @Description API for viewing Product file
|
|
// @Tags Products
|
|
// @Security Bearer
|
|
// @Param X-Client-Key header string true "Insert the X-Client-Key"
|
|
// @Param filename path string true "Product File Name (e.g., user_277788.png)"
|
|
// @Success 200 {file} file
|
|
// @Failure 400 {object} response.BadRequestError
|
|
// @Failure 401 {object} response.UnauthorizedError
|
|
// @Failure 500 {object} response.InternalServerError
|
|
// @Router /products/viewer/{filename} [get]
|
|
func (_i *productsController) Viewer(c *fiber.Ctx) error {
|
|
return _i.productsService.Viewer(c)
|
|
}
|