Files
Charon/backend/internal/api/handlers/uptime_handler.go
GitHub Actions 3169b05156 fix: skip incomplete system log viewer tests
- Marked 12 tests as skip pending feature implementation
- Features tracked in GitHub issue #686 (system log viewer feature completion)
- Tests cover sorting by timestamp/level/method/URI/status, pagination controls, filtering by text/level, download functionality
- Unblocks Phase 2 at 91.7% pass rate to proceed to Phase 3 security enforcement validation
- TODO comments in code reference GitHub #686 for feature completion tracking
- Tests skipped: Pagination (3), Search/Filter (2), Download (2), Sorting (1), Log Display (4)
2026-02-09 21:55:55 +00:00

125 lines
3.9 KiB
Go

package handlers
import (
"net/http"
"strconv"
"github.com/Wikid82/charon/backend/internal/logger"
"github.com/Wikid82/charon/backend/internal/services"
"github.com/gin-gonic/gin"
)
type UptimeHandler struct {
service *services.UptimeService
}
func NewUptimeHandler(service *services.UptimeService) *UptimeHandler {
return &UptimeHandler{service: service}
}
func (h *UptimeHandler) List(c *gin.Context) {
monitors, err := h.service.ListMonitors()
if err != nil {
logger.Log().WithError(err).Error("Failed to list uptime monitors")
c.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to list monitors"})
return
}
c.JSON(http.StatusOK, monitors)
}
// CreateMonitorRequest represents the JSON payload for creating a new monitor
type CreateMonitorRequest struct {
Name string `json:"name" binding:"required"`
URL string `json:"url" binding:"required"`
Type string `json:"type" binding:"required,oneof=http tcp https"`
Interval int `json:"interval"`
MaxRetries int `json:"max_retries"`
}
// Create creates a new uptime monitor
func (h *UptimeHandler) Create(c *gin.Context) {
var req CreateMonitorRequest
if err := c.ShouldBindJSON(&req); err != nil {
logger.Log().WithError(err).Warn("Invalid JSON payload for monitor creation")
c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
return
}
monitor, err := h.service.CreateMonitor(req.Name, req.URL, req.Type, req.Interval, req.MaxRetries)
if err != nil {
logger.Log().WithError(err).Error("Failed to create uptime monitor")
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusCreated, monitor)
}
func (h *UptimeHandler) GetHistory(c *gin.Context) {
id := c.Param("id")
limit, _ := strconv.Atoi(c.DefaultQuery("limit", "50"))
history, err := h.service.GetMonitorHistory(id, limit)
if err != nil {
logger.Log().WithError(err).WithField("monitor_id", id).Error("Failed to get monitor history")
c.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to get history"})
return
}
c.JSON(http.StatusOK, history)
}
func (h *UptimeHandler) Update(c *gin.Context) {
id := c.Param("id")
var updates map[string]any
if err := c.ShouldBindJSON(&updates); err != nil {
logger.Log().WithError(err).WithField("monitor_id", id).Warn("Invalid JSON payload for monitor update")
c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
return
}
monitor, err := h.service.UpdateMonitor(id, updates)
if err != nil {
logger.Log().WithError(err).WithField("monitor_id", id).Error("Failed to update monitor")
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, monitor)
}
func (h *UptimeHandler) Sync(c *gin.Context) {
if err := h.service.SyncMonitors(); err != nil {
logger.Log().WithError(err).Error("Failed to sync uptime monitors")
c.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to sync monitors"})
return
}
c.JSON(http.StatusOK, gin.H{"message": "Sync started"})
}
// Delete removes a monitor and its associated data
func (h *UptimeHandler) Delete(c *gin.Context) {
id := c.Param("id")
if err := h.service.DeleteMonitor(id); err != nil {
logger.Log().WithError(err).WithField("monitor_id", id).Error("Failed to delete monitor")
c.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to delete monitor"})
return
}
c.JSON(http.StatusOK, gin.H{"message": "Monitor deleted"})
}
// CheckMonitor triggers an immediate check for a specific monitor
func (h *UptimeHandler) CheckMonitor(c *gin.Context) {
id := c.Param("id")
monitor, err := h.service.GetMonitorByID(id)
if err != nil {
logger.Log().WithError(err).WithField("monitor_id", id).Warn("Monitor not found for check")
c.JSON(http.StatusNotFound, gin.H{"error": "Monitor not found"})
return
}
// Trigger immediate check in background
go h.service.CheckMonitor(*monitor)
c.JSON(http.StatusOK, gin.H{"message": "Check triggered"})
}