zdravko/internal/handlers/index.go

180 lines
4.3 KiB
Go
Raw Normal View History

package handlers
import (
2024-02-22 16:29:17 +00:00
"context"
"net/http"
2024-02-22 16:29:17 +00:00
"time"
2024-02-27 11:04:05 +00:00
"code.tjo.space/mentos1386/zdravko/database/models"
2024-02-22 16:29:17 +00:00
"code.tjo.space/mentos1386/zdravko/internal/services"
"code.tjo.space/mentos1386/zdravko/web/templates/components"
"github.com/labstack/echo/v4"
)
type IndexData struct {
*components.Base
2024-03-04 13:20:01 +00:00
Monitors map[string]MonitorsAndStatus
MonitorsLength int
2024-02-27 11:04:05 +00:00
TimeRange string
2024-02-29 22:42:56 +00:00
Status models.MonitorStatus
}
2024-03-03 14:28:25 +00:00
type Monitor struct {
2024-02-29 22:42:56 +00:00
Name string
2024-03-03 14:28:25 +00:00
Group string
2024-02-29 22:42:56 +00:00
Status models.MonitorStatus
History *History
}
2024-03-04 13:20:01 +00:00
type HistoryItem struct {
Status models.MonitorStatus
Date time.Time
}
2024-02-22 16:29:17 +00:00
type History struct {
2024-03-04 13:20:01 +00:00
List []HistoryItem
2024-02-29 22:42:56 +00:00
Uptime int
2024-02-22 16:29:17 +00:00
}
2024-03-04 13:20:01 +00:00
type MonitorsAndStatus struct {
Status models.MonitorStatus
Monitors []*Monitor
}
func getDateString(date time.Time) string {
2024-02-29 22:57:23 +00:00
return date.UTC().Format("2006-01-02T15:04:05")
2024-02-22 16:29:17 +00:00
}
2024-02-29 22:42:56 +00:00
func getHistory(history []*models.MonitorHistory, period time.Duration, buckets int) *History {
historyMap := map[string]models.MonitorStatus{}
2024-02-22 16:29:17 +00:00
numOfSuccess := 0
2024-02-22 19:20:34 +00:00
numTotal := 0
2024-02-22 16:29:17 +00:00
2024-02-29 22:42:56 +00:00
for i := 0; i < buckets; i++ {
2024-03-04 13:20:01 +00:00
dateString := getDateString(time.Now().Add(period * time.Duration(-i)).Truncate(period))
historyMap[dateString] = models.MonitorUnknown
2024-02-22 16:29:17 +00:00
}
for _, _history := range history {
2024-03-04 13:20:01 +00:00
dateString := getDateString(_history.CreatedAt.Time.Truncate(period))
2024-02-22 16:29:17 +00:00
2024-02-29 22:42:56 +00:00
// Skip if not part of the "buckets"
2024-03-04 13:20:01 +00:00
if _, ok := historyMap[dateString]; !ok {
2024-02-22 16:29:17 +00:00
continue
}
numTotal++
if _history.Status == models.MonitorSuccess {
2024-02-22 16:29:17 +00:00
numOfSuccess++
}
2024-02-29 22:42:56 +00:00
// skip if it is already set to failure
2024-03-04 13:20:01 +00:00
if historyMap[dateString] == models.MonitorFailure {
2024-02-22 16:29:17 +00:00
continue
}
2024-03-04 13:20:01 +00:00
// FIXME: This is wrong! As we can have multiple checks in dateString.
// We should look at only the newest one.
historyMap[dateString] = _history.Status
2024-02-22 16:29:17 +00:00
}
2024-03-04 13:20:01 +00:00
historyItems := make([]HistoryItem, buckets)
2024-02-29 22:42:56 +00:00
for i := 0; i < buckets; i++ {
2024-03-04 13:20:01 +00:00
date := time.Now().Add(period * time.Duration(-buckets+i+1)).Truncate(period)
datestring := getDateString(date)
historyItems[i] = HistoryItem{
Status: historyMap[datestring],
Date: date,
}
2024-02-22 16:29:17 +00:00
}
2024-02-22 19:20:34 +00:00
uptime := 0
if numTotal > 0 {
uptime = 100 * numOfSuccess / numTotal
}
2024-02-22 16:29:17 +00:00
return &History{
2024-03-04 13:20:01 +00:00
List: historyItems,
2024-02-29 22:42:56 +00:00
Uptime: uptime,
}
}
func (h *BaseHandler) Index(c echo.Context) error {
2024-02-22 16:29:17 +00:00
ctx := context.Background()
2024-02-27 11:04:05 +00:00
monitors, err := services.GetMonitors(ctx, h.db)
2024-02-22 16:29:17 +00:00
if err != nil {
return err
}
timeRange := c.QueryParam("time-range")
2024-02-29 22:42:56 +00:00
if timeRange != "48hours" && timeRange != "90days" && timeRange != "90minutes" {
2024-02-22 16:29:17 +00:00
timeRange = "90days"
}
2024-03-04 13:20:01 +00:00
overallStatus := models.MonitorUnknown
statusByGroup := make(map[string]models.MonitorStatus)
2024-02-22 16:29:17 +00:00
2024-03-03 14:28:25 +00:00
monitorsWithHistory := make([]*Monitor, len(monitors))
for i, monitor := range monitors {
2024-02-29 22:42:56 +00:00
history, err := services.GetMonitorHistoryForMonitor(ctx, h.db, monitor.Id)
2024-02-27 11:04:05 +00:00
if err != nil {
return err
}
2024-02-29 22:42:56 +00:00
var historyResult *History
switch timeRange {
case "48hours":
historyResult = getHistory(history, time.Hour, 48)
case "90days":
historyResult = getHistory(history, time.Hour*24, 90)
case "90minutes":
historyResult = getHistory(history, time.Minute, 90)
}
2024-02-22 16:29:17 +00:00
2024-03-04 13:20:01 +00:00
if statusByGroup[monitor.Group] == "" {
statusByGroup[monitor.Group] = models.MonitorUnknown
}
2024-02-29 22:42:56 +00:00
status := historyResult.List[len(historyResult.List)-1]
2024-03-04 13:20:01 +00:00
if status.Status == models.MonitorSuccess {
if overallStatus == models.MonitorUnknown {
overallStatus = status.Status
}
if statusByGroup[monitor.Group] == models.MonitorUnknown {
statusByGroup[monitor.Group] = status.Status
}
}
if status.Status != models.MonitorSuccess && status.Status != models.MonitorUnknown {
overallStatus = status.Status
statusByGroup[monitor.Group] = status.Status
2024-02-22 16:29:17 +00:00
}
2024-03-03 14:28:25 +00:00
monitorsWithHistory[i] = &Monitor{
2024-02-29 22:42:56 +00:00
Name: monitor.Name,
2024-03-03 14:28:25 +00:00
Group: monitor.Group,
2024-03-04 13:20:01 +00:00
Status: status.Status,
2024-02-29 22:42:56 +00:00
History: historyResult,
2024-02-22 16:29:17 +00:00
}
}
2024-03-04 13:20:01 +00:00
monitorsByGroup := map[string]MonitorsAndStatus{}
2024-03-03 14:28:25 +00:00
for _, monitor := range monitorsWithHistory {
2024-03-04 13:20:01 +00:00
monitorsByGroup[monitor.Group] = MonitorsAndStatus{
Status: statusByGroup[monitor.Group],
Monitors: append(monitorsByGroup[monitor.Group].Monitors, monitor),
}
2024-03-03 14:28:25 +00:00
}
c.Response().Header().Set("Cache-Control", "max-age=10")
return c.Render(http.StatusOK, "index.tmpl", &IndexData{
Base: &components.Base{
NavbarActive: GetPageByTitle(Pages, "Status"),
Navbar: Pages,
},
2024-03-03 14:28:25 +00:00
Monitors: monitorsByGroup,
TimeRange: timeRange,
Status: overallStatus,
})
}