Files
RRBEC-LocalServer/rrbec_server/internal/api/handlers.go

550 lines
14 KiB
Go

package api
import (
"log"
"net/http"
"rrbec_server/internal/models"
"rrbec_server/internal/repository"
"rrbec_server/internal/service"
"strconv"
"time"
"github.com/gin-gonic/gin"
)
type Handler struct {
svc *service.Service
syncer interface{ SyncProductsOnly() }
}
func NewHandler(svc *service.Service) *Handler {
return &Handler{svc: svc}
}
func (h *Handler) SetSyncer(syncer interface{ SyncProductsOnly() }) {
h.syncer = syncer
}
func parsePagination(c *gin.Context) *repository.PaginationParams {
page, _ := strconv.Atoi(c.DefaultQuery("page", "1"))
limit, _ := strconv.Atoi(c.DefaultQuery("limit", "50"))
sort := c.DefaultQuery("sort", "id ASC")
params := &repository.PaginationParams{
Page: page,
Limit: limit,
Sort: sort,
Filters: make(map[string]interface{}),
}
if name := c.Query("name"); name != "" {
params.Filters["name"] = name
}
if active := c.Query("active"); active != "" {
if b, err := strconv.ParseBool(active); err == nil {
params.Filters["active"] = b
}
}
if status := c.Query("status"); status != "" {
params.Filters["status"] = status
}
if category := c.Query("category"); category != "" {
if c, err := strconv.ParseUint(category, 10, 64); err == nil {
params.Filters["category"] = uint(c)
}
}
if cuisine := c.Query("cuisine"); cuisine != "" {
if b, err := strconv.ParseBool(cuisine); err == nil {
params.Filters["cuisine"] = b
}
}
if dateFrom := c.Query("date_from"); dateFrom != "" {
if t, err := time.Parse("2006-01-02", dateFrom); err == nil {
params.Filters["date_from"] = t
}
}
if dateTo := c.Query("date_to"); dateTo != "" {
if t, err := time.Parse("2006-01-02", dateTo); err == nil {
params.Filters["date_to"] = t
}
}
if comandaID := c.Query("comanda_id"); comandaID != "" {
if c, err := strconv.ParseUint(comandaID, 10, 64); err == nil {
params.Filters["comanda_id"] = uint(c)
}
}
if canceled := c.Query("canceled"); canceled != "" {
if b, err := strconv.ParseBool(canceled); err == nil {
params.Filters["canceled"] = b
}
}
if delivered := c.Query("delivered"); delivered != "" {
if b, err := strconv.ParseBool(delivered); err == nil {
params.Filters["delivered"] = b
}
}
return params
}
func (h *Handler) GetProducts(c *gin.Context) {
params := parsePagination(c)
result, err := h.svc.GetProducts(params)
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, result)
}
func (h *Handler) CreateProduct(c *gin.Context) {
var product models.Product
if err := c.ShouldBindJSON(&product); err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
return
}
if err := h.svc.CreateProduct(&product); err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusCreated, gin.H{"message": "Product created successfully", "product": product})
}
func (h *Handler) UpdateProduct(c *gin.Context) {
idStr := c.Param("id")
id, err := strconv.ParseUint(idStr, 10, 64)
if err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": "invalid id format"})
return
}
var updates map[string]interface{}
if err := c.ShouldBindJSON(&updates); err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
return
}
if err := h.svc.UpdateProduct(uint(id), updates); err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, gin.H{"message": "Product updated successfully"})
}
func (h *Handler) GetMesas(c *gin.Context) {
params := parsePagination(c)
result, err := h.svc.GetMesas(params)
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, result)
}
func (h *Handler) GetCategories(c *gin.Context) {
params := parsePagination(c)
result, err := h.svc.GetCategories(params)
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, result)
}
func (h *Handler) CreateCategory(c *gin.Context) {
var cat models.Category
if err := c.ShouldBindJSON(&cat); err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
return
}
if err := h.svc.CreateCategory(&cat); err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusCreated, cat)
}
func (h *Handler) UpdateCategory(c *gin.Context) {
idStr := c.Param("id")
id, err := strconv.ParseUint(idStr, 10, 64)
if err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": "invalid id format"})
return
}
var updates map[string]interface{}
if err := c.ShouldBindJSON(&updates); err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
return
}
if err := h.svc.UpdateCategory(uint(id), updates); err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, gin.H{"message": "Category updated successfully"})
}
func (h *Handler) GetTypePayments(c *gin.Context) {
params := parsePagination(c)
result, err := h.svc.GetTypePayments(params)
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, result)
}
func (h *Handler) GetClients(c *gin.Context) {
params := parsePagination(c)
result, err := h.svc.GetClients(params)
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, result)
}
func (h *Handler) GetOrders(c *gin.Context) {
params := parsePagination(c)
result, err := h.svc.GetOrders(params)
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, result)
}
func (h *Handler) CreateOrder(c *gin.Context) {
var order models.Order
if err := c.ShouldBindJSON(&order); err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
return
}
if err := h.svc.CreateOrder(&order); err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusCreated, order)
}
func (h *Handler) UpdateOrder(c *gin.Context) {
idStr := c.Param("id")
id, err := strconv.ParseUint(idStr, 10, 64)
if err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": "invalid id format"})
return
}
var updates map[string]interface{}
if err := c.ShouldBindJSON(&updates); err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
return
}
if err := h.svc.UpdateOrder(uint(id), updates); err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, gin.H{"message": "Order updated successfully"})
}
func (h *Handler) SetOrderPreparing(c *gin.Context) {
idStr := c.Param("id")
id, err := strconv.ParseUint(idStr, 10, 64)
if err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": "invalid id format"})
return
}
if err := h.svc.SetOrderPreparing(uint(id)); err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, gin.H{"message": "Order is now preparing"})
}
func (h *Handler) SetOrderFinished(c *gin.Context) {
idStr := c.Param("id")
id, err := strconv.ParseUint(idStr, 10, 64)
if err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": "invalid id format"})
return
}
if err := h.svc.SetOrderFinished(uint(id)); err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, gin.H{"message": "Order finished"})
}
func (h *Handler) SetOrderDelivered(c *gin.Context) {
idStr := c.Param("id")
id, err := strconv.ParseUint(idStr, 10, 64)
if err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": "invalid id format"})
return
}
if err := h.svc.SetOrderDelivered(uint(id)); err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, gin.H{"message": "Order delivered"})
}
func (h *Handler) SetOrderCanceled(c *gin.Context) {
idStr := c.Param("id")
id, err := strconv.ParseUint(idStr, 10, 64)
if err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": "invalid id format"})
return
}
if err := h.svc.SetOrderCanceled(uint(id)); err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, gin.H{"message": "Order canceled"})
}
func (h *Handler) GetPayments(c *gin.Context) {
params := parsePagination(c)
if c.Query("sort") == "" {
params.Sort = "id DESC"
}
result, err := h.svc.GetPayments(params)
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, result)
}
func (h *Handler) CreateComanda(c *gin.Context) {
var comanda models.Comanda
if err := c.ShouldBindJSON(&comanda); err != nil {
log.Printf("DEBUG: CreateComanda bind error: %v", err)
c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
return
}
log.Printf("DEBUG: CreateComanda received: MesaID=%d, UserID=%d, ClientID=%v", comanda.MesaID, comanda.UserID, comanda.ClientID)
if comanda.MesaID == 0 {
c.JSON(http.StatusBadRequest, gin.H{"error": "Mesa ID is required for sync consistency"})
return
}
if err := h.svc.CreateComanda(&comanda); err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusCreated, comanda)
}
func (h *Handler) AddItemToComanda(c *gin.Context) {
var item models.ProductComanda
if err := c.ShouldBindJSON(&item); err != nil {
log.Printf("DEBUG: AddItem bind error: %v", err)
c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
return
}
log.Printf("DEBUG: AddItem received: ComandaID=%d, ProductID=%d", item.ComandaID, item.ProductID)
if item.ComandaID == 0 || item.ProductID == 0 {
c.JSON(http.StatusBadRequest, gin.H{"error": "ComandaID and ProductID are required for sync"})
return
}
if err := h.svc.AddItemToComandaRaw(&item); err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusCreated, item)
}
func (h *Handler) DeleteItemFromComanda(c *gin.Context) {
idStr := c.Param("id")
id, err := strconv.ParseUint(idStr, 10, 64)
if err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": "invalid id format"})
return
}
if err := h.svc.DeleteItem(uint(id)); err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, gin.H{"message": "Item deleted successfully"})
}
func (h *Handler) ClearComanda(c *gin.Context) {
idStr := c.Param("id")
id, err := strconv.ParseUint(idStr, 10, 64)
if err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": "invalid id format"})
return
}
if err := h.svc.ClearComanda(uint(id)); err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, gin.H{"message": "Comanda cleared and closed"})
}
func (h *Handler) UpdateComanda(c *gin.Context) {
idStr := c.Param("id")
id, err := strconv.ParseUint(idStr, 10, 64)
if err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": "invalid id format"})
return
}
var updates map[string]interface{}
if err := c.ShouldBindJSON(&updates); err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
return
}
if err := h.svc.UpdateComanda(uint(id), updates); err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, gin.H{"message": "Comanda updated successfully"})
}
func (h *Handler) PagarComanda(c *gin.Context) {
idStr := c.Param("id")
id, err := strconv.ParseUint(idStr, 10, 64)
if err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": "invalid id format"})
return
}
var rawData map[string]interface{}
if err := c.ShouldBindJSON(&rawData); err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
return
}
log.Printf("DEBUG: PagarComanda RAW JSON: %v", rawData)
var payment models.Payment
if v, ok := rawData["value"].(float64); ok {
payment.Value = models.Price(v)
}
if v, ok := rawData["type_pay"].(float64); ok {
payment.TypePayID = uint(v)
} else if v, ok := rawData["id_type_pay"].(float64); ok {
payment.TypePayID = uint(v)
}
if v, ok := rawData["client"].(float64); ok {
clientId := uint(v)
payment.ClientID = &clientId
} else if v, ok := rawData["client_id"].(float64); ok {
clientId := uint(v)
payment.ClientID = &clientId
}
if v, ok := rawData["description"].(string); ok {
payment.Description = v
}
status := "CLOSED"
if v, ok := rawData["status"].(string); ok {
status = v
}
if err := h.svc.PagarComanda(uint(id), &payment, status); err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, gin.H{"message": "Payment registered and comanda updated to " + status})
}
func (h *Handler) GetComandas(c *gin.Context) {
params := parsePagination(c)
if c.Query("sort") == "" {
params.Sort = "id DESC"
}
result, err := h.svc.GetComandas(params)
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, result)
}
func (h *Handler) GetComandaByID(c *gin.Context) {
idStr := c.Param("id")
id, err := strconv.ParseUint(idStr, 10, 64)
if err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": "invalid id format"})
return
}
comanda, err := h.svc.GetComandaByID(uint(id))
if err != nil {
c.JSON(http.StatusNotFound, gin.H{"error": "comanda not found"})
return
}
c.JSON(http.StatusOK, comanda)
}
func (h *Handler) Login(c *gin.Context) {
var input struct {
Username string `json:"username" binding:"required"`
Password string `json:"password" binding:"required"`
}
if err := c.ShouldBindJSON(&input); err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
return
}
user, err := h.svc.Login(input.Username, input.Password)
if err != nil {
c.JSON(http.StatusUnauthorized, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, gin.H{
"message": "Login successful",
"user": user,
})
}
func (h *Handler) GetCurrentUser(c *gin.Context) {
userID, exists := c.Get("userID")
if !exists {
c.JSON(http.StatusUnauthorized, gin.H{"error": "user not authenticated"})
return
}
c.JSON(http.StatusOK, gin.H{"user_id": userID})
}
func (h *Handler) ResyncProducts(c *gin.Context) {
if h.syncer == nil {
c.JSON(http.StatusServiceUnavailable, gin.H{"error": "syncer not configured"})
return
}
go h.syncer.SyncProductsOnly()
c.JSON(http.StatusOK, gin.H{"message": "Product resync started in background"})
}