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 (
|
|
|
|
cachedServiceResults []byte
|
|
|
|
cachedServiceResultsGzipped []byte
|
|
|
|
cachedServiceResultsTimestamp time.Time
|
2019-09-05 01:37:13 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
func main() {
|
2019-12-04 22:44:35 +01:00
|
|
|
cfg := loadConfiguration()
|
2020-10-15 01:25:50 +02:00
|
|
|
resultsHandler := serviceResultsHandler
|
|
|
|
if cfg.Security != nil && cfg.Security.IsValid() {
|
|
|
|
resultsHandler = security.Handler(serviceResultsHandler, cfg.Security)
|
|
|
|
}
|
|
|
|
http.HandleFunc("/api/v1/results", resultsHandler)
|
2019-09-07 03:59:50 +02:00
|
|
|
http.HandleFunc("/health", healthHandler)
|
2020-09-06 06:55:01 +02:00
|
|
|
http.Handle("/", GzipHandler(http.FileServer(http.Dir("./static"))))
|
2019-12-04 22:44:35 +01:00
|
|
|
if cfg.Metrics {
|
2019-11-16 21:48:37 +01:00
|
|
|
http.Handle("/metrics", promhttp.Handler())
|
|
|
|
}
|
2019-10-06 04:20:36 +02:00
|
|
|
log.Println("[main][main] Listening on port 8080")
|
2020-04-15 01:20:00 +02:00
|
|
|
go watchdog.Monitor(cfg)
|
2019-10-06 04:20:36 +02:00
|
|
|
log.Fatal(http.ListenAndServe(":8080", 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-08-15 22:44:28 +02:00
|
|
|
func serviceResultsHandler(writer http.ResponseWriter, r *http.Request) {
|
2020-10-23 22:29:20 +02:00
|
|
|
if isExpired := cachedServiceResultsTimestamp.IsZero() || time.Now().Sub(cachedServiceResultsTimestamp) > cacheTTL; isExpired {
|
2020-08-15 22:44:28 +02:00
|
|
|
buffer := &bytes.Buffer{}
|
|
|
|
gzipWriter := gzip.NewWriter(buffer)
|
2020-10-23 22:29:20 +02:00
|
|
|
data, err := watchdog.GetJSONEncodedServiceResults()
|
2020-08-15 22:44:28 +02:00
|
|
|
if err != nil {
|
2020-09-05 03:31:28 +02:00
|
|
|
log.Printf("[main][serviceResultsHandler] 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()
|
|
|
|
cachedServiceResults = data
|
|
|
|
cachedServiceResultsGzipped = buffer.Bytes()
|
|
|
|
cachedServiceResultsTimestamp = time.Now()
|
|
|
|
}
|
|
|
|
var data []byte
|
|
|
|
if strings.Contains(r.Header.Get("Accept-Encoding"), "gzip") {
|
|
|
|
writer.Header().Set("Content-Encoding", "gzip")
|
|
|
|
data = cachedServiceResultsGzipped
|
|
|
|
} else {
|
|
|
|
data = cachedServiceResults
|
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
|
|
|
}
|