2020-12-30 02:22:17 +01:00
|
|
|
package controller
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"compress/gzip"
|
2021-01-28 00:25:37 +01:00
|
|
|
"encoding/json"
|
2020-12-30 02:22:17 +01:00
|
|
|
"fmt"
|
|
|
|
"log"
|
|
|
|
"net/http"
|
2021-01-24 10:48:07 +01:00
|
|
|
"os"
|
2020-12-30 02:22:17 +01:00
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/TwinProduction/gatus/config"
|
|
|
|
"github.com/TwinProduction/gatus/security"
|
|
|
|
"github.com/TwinProduction/gatus/watchdog"
|
2020-12-30 07:08:20 +01:00
|
|
|
"github.com/TwinProduction/gocache"
|
2020-12-30 02:22:17 +01:00
|
|
|
"github.com/gorilla/mux"
|
|
|
|
"github.com/prometheus/client_golang/prometheus/promhttp"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
cacheTTL = 10 * time.Second
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2020-12-30 07:08:20 +01:00
|
|
|
cache = gocache.NewCache().WithMaxSize(100).WithEvictionPolicy(gocache.LeastRecentlyUsed)
|
2020-12-30 02:22:17 +01:00
|
|
|
)
|
|
|
|
|
2020-12-30 07:08:20 +01:00
|
|
|
func init() {
|
|
|
|
if err := cache.StartJanitor(); err != nil {
|
|
|
|
log.Fatal("[controller][init] Failed to start cache janitor:", err.Error())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-30 02:22:17 +01:00
|
|
|
// Handle creates the router and starts the server
|
|
|
|
func Handle() {
|
|
|
|
cfg := config.Get()
|
2021-01-24 10:48:07 +01:00
|
|
|
var router http.Handler = CreateRouter(cfg)
|
|
|
|
if os.Getenv("ENVIRONMENT") == "dev" {
|
|
|
|
router = developmentCorsHandler(router)
|
|
|
|
}
|
2020-12-30 02:22:17 +01:00
|
|
|
server := &http.Server{
|
|
|
|
Addr: fmt.Sprintf("%s:%d", cfg.Web.Address, cfg.Web.Port),
|
|
|
|
Handler: router,
|
|
|
|
ReadTimeout: 15 * time.Second,
|
|
|
|
WriteTimeout: 15 * time.Second,
|
|
|
|
IdleTimeout: 15 * time.Second,
|
|
|
|
}
|
|
|
|
log.Printf("[controller][Handle] Listening on %s%s\n", cfg.Web.SocketAddress(), cfg.Web.ContextRoot)
|
|
|
|
log.Fatal(server.ListenAndServe())
|
|
|
|
}
|
|
|
|
|
|
|
|
// CreateRouter creates the router for the http server
|
|
|
|
func CreateRouter(cfg *config.Config) *mux.Router {
|
|
|
|
router := mux.NewRouter()
|
|
|
|
router.HandleFunc("/favicon.ico", favIconHandler).Methods("GET") // favicon needs to be always served from the root
|
2021-01-29 05:25:29 +01:00
|
|
|
router.HandleFunc("/services/{service}", spaHandler).Methods("GET")
|
2021-01-28 00:25:37 +01:00
|
|
|
router.HandleFunc(cfg.Web.PrependWithContextRoot("/api/v1/statuses"), secureIfNecessary(cfg, serviceStatusesHandler)).Methods("GET")
|
|
|
|
router.HandleFunc(cfg.Web.PrependWithContextRoot("/api/v1/statuses/{key}"), secureIfNecessary(cfg, GzipHandlerFunc(serviceStatusHandler))).Methods("GET")
|
2020-12-30 02:22:17 +01:00
|
|
|
router.HandleFunc(cfg.Web.PrependWithContextRoot("/api/v1/badges/uptime/{duration}/{identifier}"), badgeHandler).Methods("GET")
|
|
|
|
router.HandleFunc(cfg.Web.PrependWithContextRoot("/health"), healthHandler).Methods("GET")
|
2021-01-29 05:25:29 +01:00
|
|
|
router.PathPrefix(cfg.Web.ContextRoot).Handler(GzipHandler(http.StripPrefix(cfg.Web.ContextRoot, http.FileServer(http.Dir("./web/static")))))
|
2020-12-30 02:22:17 +01:00
|
|
|
if cfg.Metrics {
|
|
|
|
router.Handle(cfg.Web.PrependWithContextRoot("/metrics"), promhttp.Handler()).Methods("GET")
|
|
|
|
}
|
|
|
|
return router
|
|
|
|
}
|
|
|
|
|
2021-01-28 00:25:37 +01:00
|
|
|
func secureIfNecessary(cfg *config.Config, handler http.HandlerFunc) http.HandlerFunc {
|
|
|
|
if cfg.Security != nil && cfg.Security.IsValid() {
|
|
|
|
return security.Handler(serviceStatusesHandler, cfg.Security)
|
|
|
|
}
|
|
|
|
return handler
|
|
|
|
}
|
|
|
|
|
|
|
|
// serviceStatusesHandler handles requests to retrieve all service statuses
|
|
|
|
// Due to the size of the response, this function leverages a cache.
|
|
|
|
// Must not be wrapped by GzipHandler
|
2020-12-30 02:22:17 +01:00
|
|
|
func serviceStatusesHandler(writer http.ResponseWriter, r *http.Request) {
|
2020-12-30 07:08:20 +01:00
|
|
|
gzipped := strings.Contains(r.Header.Get("Accept-Encoding"), "gzip")
|
|
|
|
var exists bool
|
|
|
|
var value interface{}
|
|
|
|
if gzipped {
|
|
|
|
writer.Header().Set("Content-Encoding", "gzip")
|
|
|
|
value, exists = cache.Get("service-status-gzipped")
|
|
|
|
} else {
|
|
|
|
value, exists = cache.Get("service-status")
|
|
|
|
}
|
|
|
|
var data []byte
|
|
|
|
if !exists {
|
|
|
|
var err error
|
2020-12-30 02:22:17 +01:00
|
|
|
buffer := &bytes.Buffer{}
|
|
|
|
gzipWriter := gzip.NewWriter(buffer)
|
2020-12-31 21:28:57 +01:00
|
|
|
data, err = watchdog.GetServiceStatusesAsJSON()
|
2020-12-30 02:22:17 +01:00
|
|
|
if err != nil {
|
2021-01-28 00:25:37 +01:00
|
|
|
log.Printf("[controller][serviceStatusesHandler] Unable to marshal object to JSON: %s", err.Error())
|
2020-12-30 02:22:17 +01:00
|
|
|
writer.WriteHeader(http.StatusInternalServerError)
|
|
|
|
_, _ = writer.Write([]byte("Unable to marshal object to JSON"))
|
|
|
|
return
|
|
|
|
}
|
2020-12-30 07:08:20 +01:00
|
|
|
_, _ = gzipWriter.Write(data)
|
|
|
|
_ = gzipWriter.Close()
|
|
|
|
gzippedData := buffer.Bytes()
|
|
|
|
cache.SetWithTTL("service-status", data, cacheTTL)
|
|
|
|
cache.SetWithTTL("service-status-gzipped", gzippedData, cacheTTL)
|
|
|
|
if gzipped {
|
|
|
|
data = gzippedData
|
|
|
|
}
|
2020-12-30 02:22:17 +01:00
|
|
|
} else {
|
2020-12-30 07:08:20 +01:00
|
|
|
data = value.([]byte)
|
2020-12-30 02:22:17 +01:00
|
|
|
}
|
2020-12-30 07:08:20 +01:00
|
|
|
writer.Header().Add("Content-Type", "application/json")
|
2020-12-30 02:22:17 +01:00
|
|
|
writer.WriteHeader(http.StatusOK)
|
|
|
|
_, _ = writer.Write(data)
|
|
|
|
}
|
|
|
|
|
2021-01-28 00:25:37 +01:00
|
|
|
// serviceStatusHandler retrieves a single ServiceStatus by group name and service name
|
|
|
|
func serviceStatusHandler(writer http.ResponseWriter, r *http.Request) {
|
|
|
|
vars := mux.Vars(r)
|
|
|
|
serviceStatus := watchdog.GetServiceStatusByKey(vars["key"])
|
|
|
|
if serviceStatus == nil {
|
|
|
|
log.Printf("[controller][serviceStatusHandler] Service with key=%s not found", vars["key"])
|
|
|
|
writer.WriteHeader(http.StatusNotFound)
|
|
|
|
_, _ = writer.Write([]byte("not found"))
|
|
|
|
return
|
|
|
|
}
|
2021-01-29 04:44:31 +01:00
|
|
|
data := map[string]interface{}{
|
|
|
|
"serviceStatus": serviceStatus,
|
|
|
|
// This is my lazy way of exposing events even though they're not visible from the json annotation
|
|
|
|
// present in ServiceStatus. We do this because creating a separate object for each endpoints
|
|
|
|
// would be wasteful (one with and one without Events)
|
|
|
|
"events": serviceStatus.Events,
|
|
|
|
}
|
|
|
|
output, err := json.Marshal(data)
|
2021-01-28 00:25:37 +01:00
|
|
|
if err != nil {
|
|
|
|
log.Printf("[controller][serviceStatusHandler] Unable to marshal object to JSON: %s", err.Error())
|
|
|
|
writer.WriteHeader(http.StatusInternalServerError)
|
|
|
|
_, _ = writer.Write([]byte("Unable to marshal object to JSON"))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
writer.Header().Add("Content-Type", "application/json")
|
|
|
|
writer.WriteHeader(http.StatusOK)
|
2021-01-29 04:44:31 +01:00
|
|
|
_, _ = writer.Write(output)
|
2021-01-28 00:25:37 +01:00
|
|
|
}
|
|
|
|
|
2020-12-30 02:22:17 +01:00
|
|
|
func healthHandler(writer http.ResponseWriter, _ *http.Request) {
|
2020-12-30 07:08:20 +01:00
|
|
|
writer.Header().Add("Content-Type", "application/json")
|
2020-12-30 02:22:17 +01:00
|
|
|
writer.WriteHeader(http.StatusOK)
|
|
|
|
_, _ = writer.Write([]byte("{\"status\":\"UP\"}"))
|
|
|
|
}
|
|
|
|
|
|
|
|
// favIconHandler handles requests for /favicon.ico
|
|
|
|
func favIconHandler(writer http.ResponseWriter, request *http.Request) {
|
2021-01-29 05:25:29 +01:00
|
|
|
http.ServeFile(writer, request, "./web/static/favicon.ico")
|
|
|
|
}
|
|
|
|
|
|
|
|
// spaHandler handles requests for /favicon.ico
|
|
|
|
func spaHandler(writer http.ResponseWriter, request *http.Request) {
|
|
|
|
http.ServeFile(writer, request, "./web/static/index.html")
|
2020-12-30 02:22:17 +01:00
|
|
|
}
|