2019-09-05 01:37:13 +02:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2020-08-15 22:44:28 +02:00
|
|
|
"bytes"
|
|
|
|
"compress/gzip"
|
2019-11-16 21:48:37 +01:00
|
|
|
"log"
|
|
|
|
"net/http"
|
2019-12-04 22:44:35 +01:00
|
|
|
"os"
|
2020-08-15 22:44:28 +02:00
|
|
|
"strings"
|
|
|
|
"time"
|
2020-10-30 16:30:03 +01:00
|
|
|
|
|
|
|
"github.com/TwinProduction/gatus/config"
|
|
|
|
"github.com/TwinProduction/gatus/security"
|
|
|
|
"github.com/TwinProduction/gatus/watchdog"
|
|
|
|
"github.com/prometheus/client_golang/prometheus/promhttp"
|
2020-08-15 22:44:28 +02:00
|
|
|
)
|
|
|
|
|
2020-10-23 22:29:20 +02:00
|
|
|
const cacheTTL = 10 * time.Second
|
2020-08-15 22:44:28 +02:00
|
|
|
|
|
|
|
var (
|
2020-11-27 00:09:01 +01:00
|
|
|
cachedServiceStatuses []byte
|
|
|
|
cachedServiceStatusesGzipped []byte
|
|
|
|
cachedServiceStatusesTimestamp time.Time
|
2019-09-05 01:37:13 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
func main() {
|
2019-12-04 22:44:35 +01:00
|
|
|
cfg := loadConfiguration()
|
2020-11-27 00:09:01 +01:00
|
|
|
statusesHandler := serviceStatusesHandler
|
2020-10-15 01:25:50 +02:00
|
|
|
if cfg.Security != nil && cfg.Security.IsValid() {
|
2020-11-27 00:09:01 +01:00
|
|
|
statusesHandler = security.Handler(serviceStatusesHandler, cfg.Security)
|
2020-10-15 01:25:50 +02:00
|
|
|
}
|
2020-11-21 23:53:26 +01:00
|
|
|
http.HandleFunc("/favicon.ico", favIconHandler) // favicon needs to be always served from the root
|
2020-11-27 00:09:01 +01:00
|
|
|
http.HandleFunc(cfg.Web.PrependWithContextRoot("/api/v1/statuses"), statusesHandler)
|
2020-11-21 21:38:45 +01:00
|
|
|
http.HandleFunc(cfg.Web.PrependWithContextRoot("/health"), healthHandler)
|
2020-11-21 03:42:42 +01:00
|
|
|
http.Handle(cfg.Web.ContextRoot, GzipHandler(http.StripPrefix(cfg.Web.ContextRoot, http.FileServer(http.Dir("./static")))))
|
2020-11-21 01:44:05 +01:00
|
|
|
|
2019-12-04 22:44:35 +01:00
|
|
|
if cfg.Metrics {
|
2020-11-21 21:38:45 +01:00
|
|
|
http.Handle(cfg.Web.PrependWithContextRoot("/metrics"), promhttp.Handler())
|
2019-11-16 21:48:37 +01:00
|
|
|
}
|
2020-11-21 03:42:42 +01:00
|
|
|
log.Printf("[main][main] Listening on %s%s\n", cfg.Web.SocketAddress(), cfg.Web.ContextRoot)
|
2020-04-15 01:20:00 +02:00
|
|
|
go watchdog.Monitor(cfg)
|
2020-11-20 23:40:57 +01:00
|
|
|
log.Fatal(http.ListenAndServe(cfg.Web.SocketAddress(), nil))
|
2019-09-07 03:59:50 +02:00
|
|
|
}
|
|
|
|
|
2019-12-04 22:44:35 +01:00
|
|
|
func loadConfiguration() *config.Config {
|
|
|
|
var err error
|
2020-03-08 23:16:39 +01:00
|
|
|
customConfigFile := os.Getenv("GATUS_CONFIG_FILE")
|
|
|
|
if len(customConfigFile) > 0 {
|
|
|
|
err = config.Load(customConfigFile)
|
2019-12-04 22:44:35 +01:00
|
|
|
} else {
|
|
|
|
err = config.LoadDefaultConfiguration()
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return config.Get()
|
|
|
|
}
|
|
|
|
|
2020-11-27 00:09:01 +01:00
|
|
|
func serviceStatusesHandler(writer http.ResponseWriter, r *http.Request) {
|
|
|
|
if isExpired := cachedServiceStatusesTimestamp.IsZero() || time.Now().Sub(cachedServiceStatusesTimestamp) > cacheTTL; isExpired {
|
2020-08-15 22:44:28 +02:00
|
|
|
buffer := &bytes.Buffer{}
|
|
|
|
gzipWriter := gzip.NewWriter(buffer)
|
2020-11-27 00:09:01 +01:00
|
|
|
data, err := watchdog.GetJSONEncodedServiceStatuses()
|
2020-08-15 22:44:28 +02:00
|
|
|
if err != nil {
|
2020-11-27 00:09:01 +01:00
|
|
|
log.Printf("[main][serviceStatusesHandler] Unable to marshal object to JSON: %s", err.Error())
|
2020-08-15 22:44:28 +02:00
|
|
|
writer.WriteHeader(http.StatusInternalServerError)
|
2020-09-05 03:31:28 +02:00
|
|
|
_, _ = writer.Write([]byte("Unable to marshal object to JSON"))
|
2020-08-15 22:44:28 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
gzipWriter.Write(data)
|
|
|
|
gzipWriter.Close()
|
2020-11-27 00:09:01 +01:00
|
|
|
cachedServiceStatuses = data
|
|
|
|
cachedServiceStatusesGzipped = buffer.Bytes()
|
|
|
|
cachedServiceStatusesTimestamp = time.Now()
|
2020-08-15 22:44:28 +02:00
|
|
|
}
|
|
|
|
var data []byte
|
|
|
|
if strings.Contains(r.Header.Get("Accept-Encoding"), "gzip") {
|
|
|
|
writer.Header().Set("Content-Encoding", "gzip")
|
2020-11-27 00:09:01 +01:00
|
|
|
data = cachedServiceStatusesGzipped
|
2020-08-15 22:44:28 +02:00
|
|
|
} else {
|
2020-11-27 00:09:01 +01:00
|
|
|
data = cachedServiceStatuses
|
2020-04-15 01:20:00 +02:00
|
|
|
}
|
2019-12-28 18:19:52 +01:00
|
|
|
writer.Header().Add("Content-type", "application/json")
|
2019-09-07 03:59:50 +02:00
|
|
|
writer.WriteHeader(http.StatusOK)
|
2020-04-15 01:20:00 +02:00
|
|
|
_, _ = writer.Write(data)
|
2019-09-07 03:59:50 +02:00
|
|
|
}
|
|
|
|
|
2019-11-16 21:48:37 +01:00
|
|
|
func healthHandler(writer http.ResponseWriter, _ *http.Request) {
|
2019-12-28 18:19:52 +01:00
|
|
|
writer.Header().Add("Content-type", "application/json")
|
2019-09-07 03:59:50 +02:00
|
|
|
writer.WriteHeader(http.StatusOK)
|
2020-04-15 01:20:00 +02:00
|
|
|
_, _ = writer.Write([]byte("{\"status\":\"UP\"}"))
|
2019-09-05 01:37:13 +02:00
|
|
|
}
|
2020-11-21 01:44:05 +01:00
|
|
|
|
2020-11-21 23:53:26 +01:00
|
|
|
// favIconHandler handles requests for /favicon.ico
|
2020-11-21 01:44:05 +01:00
|
|
|
func favIconHandler(writer http.ResponseWriter, request *http.Request) {
|
|
|
|
http.ServeFile(writer, request, "./static/favicon.ico")
|
|
|
|
}
|