This will allow importing storage.Config without importing every SQL drivers in the known universe
		
			
				
	
	
		
			104 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			104 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package handler
 | |
| 
 | |
| import (
 | |
| 	"bytes"
 | |
| 	"compress/gzip"
 | |
| 	"encoding/json"
 | |
| 	"fmt"
 | |
| 	"log"
 | |
| 	"net/http"
 | |
| 	"strings"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/TwiN/gatus/v3/storage/store"
 | |
| 	"github.com/TwiN/gatus/v3/storage/store/common"
 | |
| 	"github.com/TwiN/gatus/v3/storage/store/common/paging"
 | |
| 	"github.com/TwiN/gocache"
 | |
| 	"github.com/gorilla/mux"
 | |
| )
 | |
| 
 | |
| const (
 | |
| 	cacheTTL = 10 * time.Second
 | |
| )
 | |
| 
 | |
| var (
 | |
| 	cache = gocache.NewCache().WithMaxSize(100).WithEvictionPolicy(gocache.FirstInFirstOut)
 | |
| )
 | |
| 
 | |
| // EndpointStatuses handles requests to retrieve all EndpointStatus
 | |
| // Due to the size of the response, this function leverages a cache.
 | |
| // Must not be wrapped by GzipHandler
 | |
| func EndpointStatuses(writer http.ResponseWriter, r *http.Request) {
 | |
| 	page, pageSize := extractPageAndPageSizeFromRequest(r)
 | |
| 	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(fmt.Sprintf("endpoint-status-%d-%d-gzipped", page, pageSize))
 | |
| 	} else {
 | |
| 		value, exists = cache.Get(fmt.Sprintf("endpoint-status-%d-%d", page, pageSize))
 | |
| 	}
 | |
| 	var data []byte
 | |
| 	if !exists {
 | |
| 		var err error
 | |
| 		buffer := &bytes.Buffer{}
 | |
| 		gzipWriter := gzip.NewWriter(buffer)
 | |
| 		endpointStatuses, err := store.Get().GetAllEndpointStatuses(paging.NewEndpointStatusParams().WithResults(page, pageSize))
 | |
| 		if err != nil {
 | |
| 			log.Printf("[handler][EndpointStatuses] Failed to retrieve endpoint statuses: %s", err.Error())
 | |
| 			http.Error(writer, err.Error(), http.StatusInternalServerError)
 | |
| 			return
 | |
| 		}
 | |
| 		data, err = json.Marshal(endpointStatuses)
 | |
| 		if err != nil {
 | |
| 			log.Printf("[handler][EndpointStatuses] Unable to marshal object to JSON: %s", err.Error())
 | |
| 			http.Error(writer, "unable to marshal object to JSON", http.StatusInternalServerError)
 | |
| 			return
 | |
| 		}
 | |
| 		_, _ = gzipWriter.Write(data)
 | |
| 		_ = gzipWriter.Close()
 | |
| 		gzippedData := buffer.Bytes()
 | |
| 		cache.SetWithTTL(fmt.Sprintf("endpoint-status-%d-%d", page, pageSize), data, cacheTTL)
 | |
| 		cache.SetWithTTL(fmt.Sprintf("endpoint-status-%d-%d-gzipped", page, pageSize), gzippedData, cacheTTL)
 | |
| 		if gzipped {
 | |
| 			data = gzippedData
 | |
| 		}
 | |
| 	} else {
 | |
| 		data = value.([]byte)
 | |
| 	}
 | |
| 	writer.Header().Add("Content-Type", "application/json")
 | |
| 	writer.WriteHeader(http.StatusOK)
 | |
| 	_, _ = writer.Write(data)
 | |
| }
 | |
| 
 | |
| // EndpointStatus retrieves a single core.EndpointStatus by group and endpoint name
 | |
| func EndpointStatus(writer http.ResponseWriter, r *http.Request) {
 | |
| 	page, pageSize := extractPageAndPageSizeFromRequest(r)
 | |
| 	vars := mux.Vars(r)
 | |
| 	endpointStatus, err := store.Get().GetEndpointStatusByKey(vars["key"], paging.NewEndpointStatusParams().WithResults(page, pageSize).WithEvents(1, common.MaximumNumberOfEvents))
 | |
| 	if err != nil {
 | |
| 		if err == common.ErrEndpointNotFound {
 | |
| 			http.Error(writer, err.Error(), http.StatusNotFound)
 | |
| 			return
 | |
| 		}
 | |
| 		log.Printf("[handler][EndpointStatus] Failed to retrieve endpoint status: %s", err.Error())
 | |
| 		http.Error(writer, err.Error(), http.StatusInternalServerError)
 | |
| 		return
 | |
| 	}
 | |
| 	if endpointStatus == nil {
 | |
| 		log.Printf("[handler][EndpointStatus] Endpoint with key=%s not found", vars["key"])
 | |
| 		http.Error(writer, "not found", http.StatusNotFound)
 | |
| 		return
 | |
| 	}
 | |
| 	output, err := json.Marshal(endpointStatus)
 | |
| 	if err != nil {
 | |
| 		log.Printf("[handler][EndpointStatus] Unable to marshal object to JSON: %s", err.Error())
 | |
| 		http.Error(writer, "unable to marshal object to JSON", http.StatusInternalServerError)
 | |
| 		return
 | |
| 	}
 | |
| 	writer.Header().Add("Content-Type", "application/json")
 | |
| 	writer.WriteHeader(http.StatusOK)
 | |
| 	_, _ = writer.Write(output)
 | |
| }
 |