2019-09-05 01:37:13 +02:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2020-08-15 22:44:28 +02:00
|
|
|
"bytes"
|
|
|
|
"compress/gzip"
|
2019-09-07 03:59:50 +02:00
|
|
|
"encoding/json"
|
2019-11-16 21:48:37 +01:00
|
|
|
"github.com/TwinProduction/gatus/config"
|
2019-10-06 04:20:36 +02:00
|
|
|
"github.com/TwinProduction/gatus/watchdog"
|
2019-11-16 21:48:37 +01:00
|
|
|
"github.com/prometheus/client_golang/prometheus/promhttp"
|
|
|
|
"log"
|
|
|
|
"net/http"
|
2019-12-04 22:44:35 +01:00
|
|
|
"os"
|
2020-08-15 22:44:28 +02:00
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
const CacheTTL = 10 * time.Second
|
|
|
|
|
|
|
|
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()
|
2019-09-07 03:59:50 +02:00
|
|
|
http.HandleFunc("/api/v1/results", serviceResultsHandler)
|
|
|
|
http.HandleFunc("/health", healthHandler)
|
2019-09-12 22:15:42 +02:00
|
|
|
http.Handle("/", 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) {
|
|
|
|
if isExpired := cachedServiceResultsTimestamp.IsZero() || time.Now().Sub(cachedServiceResultsTimestamp) > CacheTTL; isExpired {
|
|
|
|
buffer := &bytes.Buffer{}
|
|
|
|
gzipWriter := gzip.NewWriter(buffer)
|
|
|
|
serviceResults := watchdog.GetServiceResults()
|
|
|
|
data, err := json.Marshal(serviceResults)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("[main][serviceResultsHandler] Unable to marshall object to JSON: %s", err.Error())
|
|
|
|
writer.WriteHeader(http.StatusInternalServerError)
|
|
|
|
_, _ = writer.Write([]byte("Unable to marshall object to JSON"))
|
|
|
|
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
|
|
|
}
|