photum/backend/internal/escalas/handler.go
NANDO9322 d471b4fc0d - Adiciona filtro de role RESEARCHER na tela de Aprovação.
- Implementa edição de Role na tela de Aprovação com suporte a funções virtuais (Cine/Recep).
- Atualiza apiService com updateUserRole.
- Corrige visibilidade do Dashboard para RESEARCHER (DataContext).
- Backend: ListPending retorna tipo_profissional original.
2026-01-31 14:20:51 -03:00

180 lines
4.8 KiB
Go

package escalas
import (
"net/http"
"github.com/gin-gonic/gin"
"github.com/google/uuid"
)
type Handler struct {
service *Service
}
func NewHandler(service *Service) *Handler {
return &Handler{service: service}
}
// Create godoc
// @Summary Create a schedule entry
// @Description Assign a professional to a time block in an event
// @Tags escalas
// @Accept json
// @Produce json
// @Param request body CreateEscalaInput true "Create Escala"
// @Success 201 {object} map[string]string
// @Router /api/escalas [post]
func (h *Handler) Create(c *gin.Context) {
var req CreateEscalaInput
if err := c.ShouldBindJSON(&req); err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
return
}
// Security: Block RESEARCHER
if c.GetString("role") == "RESEARCHER" {
c.JSON(http.StatusForbidden, gin.H{"error": "Acesso negado: Somente leitura"})
return
}
escala, err := h.service.Create(c.Request.Context(), req)
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusCreated, gin.H{"id": uuid.UUID(escala.ID.Bytes).String()})
}
// ListByAgenda godoc
// @Summary List schedules for an event
// @Description Get all schedule entries for a specific agenda_id
// @Tags escalas
// @Accept json
// @Produce json
// @Param agenda_id query string true "Agenda ID"
// @Success 200 {array} map[string]interface{}
// @Router /api/escalas [get]
func (h *Handler) ListByAgenda(c *gin.Context) {
agendaID := c.Query("agenda_id")
if agendaID == "" {
c.JSON(http.StatusBadRequest, gin.H{"error": "agenda_id is required"})
return
}
escalas, err := h.service.ListByAgenda(c.Request.Context(), agendaID)
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
resp := make([]map[string]interface{}, len(escalas))
for i, e := range escalas {
var funcao string
if e.FuncaoEspecifica.Valid {
funcao = e.FuncaoEspecifica.String
}
var profNome string
profNome = e.ProfissionalNome
var avatar string
if e.AvatarUrl.Valid {
avatar = e.AvatarUrl.String
}
var phone string
if e.Whatsapp.Valid {
if c.GetString("role") == "RESEARCHER" {
phone = "Bloqueado"
} else {
phone = e.Whatsapp.String
}
}
var profFuncao string
if e.FuncaoNome.Valid {
profFuncao = e.FuncaoNome.String
}
resp[i] = map[string]interface{}{
"id": uuid.UUID(e.ID.Bytes).String(),
"agenda_id": uuid.UUID(e.AgendaID.Bytes).String(),
"profissional_id": uuid.UUID(e.ProfissionalID.Bytes).String(),
"profissional_nome": profNome,
"avatar_url": avatar,
"phone": phone,
"profissional_role": profFuncao,
"start": e.DataHoraInicio.Time,
"end": e.DataHoraFim.Time,
"role": funcao, // Specific role in this slot
}
}
c.JSON(http.StatusOK, resp)
}
// Delete godoc
// @Summary Delete a schedule entry
// @Description Remove a professional from a time block
// @Tags escalas
// @Accept json
// @Produce json
// @Param id path string true "Escala ID"
// @Success 200 {object} map[string]string
// @Router /api/escalas/{id} [delete]
func (h *Handler) Delete(c *gin.Context) {
id := c.Param("id")
if id == "" {
c.JSON(http.StatusBadRequest, gin.H{"error": "id required"})
return
}
// Security: Block RESEARCHER
if c.GetString("role") == "RESEARCHER" {
c.JSON(http.StatusForbidden, gin.H{"error": "Acesso negado: Somente leitura"})
return
}
err := h.service.Delete(c.Request.Context(), id)
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, gin.H{"message": "deleted"})
}
// Update godoc
// @Summary Update a schedule entry
// @Description Update time or professional for a slot
// @Tags escalas
// @Accept json
// @Produce json
// @Param id path string true "Escala ID"
// @Param request body UpdateEscalaInput true "Update Escala"
// @Success 200 {object} map[string]string
// @Router /api/escalas/{id} [put]
func (h *Handler) Update(c *gin.Context) {
id := c.Param("id")
if id == "" {
c.JSON(http.StatusBadRequest, gin.H{"error": "id required"})
return
}
var req UpdateEscalaInput
if err := c.ShouldBindJSON(&req); err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
return
}
// Security: Block RESEARCHER
if c.GetString("role") == "RESEARCHER" {
c.JSON(http.StatusForbidden, gin.H{"error": "Acesso negado: Somente leitura"})
return
}
_, err := h.service.Update(c.Request.Context(), id, req)
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, gin.H{"message": "updated"})
}