mirror of
https://github.com/TwiN/gatus.git
synced 2024-12-23 15:18:55 +01:00
268 lines
8.2 KiB
Go
268 lines
8.2 KiB
Go
package core
|
|
|
|
import (
|
|
"bytes"
|
|
"crypto/x509"
|
|
"encoding/json"
|
|
"errors"
|
|
"io/ioutil"
|
|
"net"
|
|
"net/http"
|
|
"net/url"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/TwinProduction/gatus/alerting/alert"
|
|
"github.com/TwinProduction/gatus/client"
|
|
"github.com/TwinProduction/gatus/util"
|
|
)
|
|
|
|
const (
|
|
// HostHeader is the name of the header used to specify the host
|
|
HostHeader = "Host"
|
|
|
|
// ContentTypeHeader is the name of the header used to specify the content type
|
|
ContentTypeHeader = "Content-Type"
|
|
|
|
// UserAgentHeader is the name of the header used to specify the request's user agent
|
|
UserAgentHeader = "User-Agent"
|
|
|
|
// GatusUserAgent is the default user agent that Gatus uses to send requests.
|
|
GatusUserAgent = "Gatus/1.0"
|
|
)
|
|
|
|
var (
|
|
// ErrServiceWithNoCondition is the error with which Gatus will panic if a service is configured with no conditions
|
|
ErrServiceWithNoCondition = errors.New("you must specify at least one condition per service")
|
|
|
|
// ErrServiceWithNoURL is the error with which Gatus will panic if a service is configured with no url
|
|
ErrServiceWithNoURL = errors.New("you must specify an url for each service")
|
|
|
|
// ErrServiceWithNoName is the error with which Gatus will panic if a service is configured with no name
|
|
ErrServiceWithNoName = errors.New("you must specify a name for each service")
|
|
)
|
|
|
|
// Service is the configuration of a monitored endpoint
|
|
type Service struct {
|
|
// Name of the service. Can be anything.
|
|
Name string `yaml:"name"`
|
|
|
|
// Group the service is a part of. Used for grouping multiple services together on the front end.
|
|
Group string `yaml:"group,omitempty"`
|
|
|
|
// URL to send the request to
|
|
URL string `yaml:"url"`
|
|
|
|
// DNS is the configuration of DNS monitoring
|
|
DNS *DNS `yaml:"dns,omitempty"`
|
|
|
|
// Method of the request made to the url of the service
|
|
Method string `yaml:"method,omitempty"`
|
|
|
|
// Body of the request
|
|
Body string `yaml:"body,omitempty"`
|
|
|
|
// GraphQL is whether to wrap the body in a query param ({"query":"$body"})
|
|
GraphQL bool `yaml:"graphql,omitempty"`
|
|
|
|
// Headers of the request
|
|
Headers map[string]string `yaml:"headers,omitempty"`
|
|
|
|
// Interval is the duration to wait between every status check
|
|
Interval time.Duration `yaml:"interval,omitempty"`
|
|
|
|
// Conditions used to determine the health of the service
|
|
Conditions []*Condition `yaml:"conditions"`
|
|
|
|
// Alerts is the alerting configuration for the service in case of failure
|
|
Alerts []*alert.Alert `yaml:"alerts"`
|
|
|
|
// Insecure is whether to skip verifying the server's certificate chain and host name
|
|
Insecure bool `yaml:"insecure,omitempty"`
|
|
|
|
// NumberOfFailuresInARow is the number of unsuccessful evaluations in a row
|
|
NumberOfFailuresInARow int
|
|
|
|
// NumberOfSuccessesInARow is the number of successful evaluations in a row
|
|
NumberOfSuccessesInARow int
|
|
}
|
|
|
|
// ValidateAndSetDefaults validates the service's configuration and sets the default value of fields that have one
|
|
func (service *Service) ValidateAndSetDefaults() error {
|
|
// Set default values
|
|
if service.Interval == 0 {
|
|
service.Interval = 1 * time.Minute
|
|
}
|
|
if len(service.Method) == 0 {
|
|
service.Method = http.MethodGet
|
|
}
|
|
if len(service.Headers) == 0 {
|
|
service.Headers = make(map[string]string)
|
|
}
|
|
// Automatically add user agent header if there isn't one specified in the service configuration
|
|
if _, userAgentHeaderExists := service.Headers[UserAgentHeader]; !userAgentHeaderExists {
|
|
service.Headers[UserAgentHeader] = GatusUserAgent
|
|
}
|
|
// Automatically add "Content-Type: application/json" header if there's no Content-Type set
|
|
// and service.GraphQL is set to true
|
|
if _, contentTypeHeaderExists := service.Headers[ContentTypeHeader]; !contentTypeHeaderExists && service.GraphQL {
|
|
service.Headers[ContentTypeHeader] = "application/json"
|
|
}
|
|
for _, serviceAlert := range service.Alerts {
|
|
if serviceAlert.FailureThreshold <= 0 {
|
|
serviceAlert.FailureThreshold = 3
|
|
}
|
|
if serviceAlert.SuccessThreshold <= 0 {
|
|
serviceAlert.SuccessThreshold = 2
|
|
}
|
|
}
|
|
if len(service.Name) == 0 {
|
|
return ErrServiceWithNoName
|
|
}
|
|
if len(service.URL) == 0 {
|
|
return ErrServiceWithNoURL
|
|
}
|
|
if len(service.Conditions) == 0 {
|
|
return ErrServiceWithNoCondition
|
|
}
|
|
if service.DNS != nil {
|
|
return service.DNS.validateAndSetDefault()
|
|
}
|
|
// Make sure that the request can be created
|
|
_, err := http.NewRequest(service.Method, service.URL, bytes.NewBuffer([]byte(service.Body)))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Key returns the unique key for the Service
|
|
func (service Service) Key() string {
|
|
return util.ConvertGroupAndServiceToKey(service.Group, service.Name)
|
|
}
|
|
|
|
// EvaluateHealth sends a request to the service's URL and evaluates the conditions of the service.
|
|
func (service *Service) EvaluateHealth() *Result {
|
|
result := &Result{Success: true, Errors: []string{}}
|
|
service.getIP(result)
|
|
if len(result.Errors) == 0 {
|
|
service.call(result)
|
|
} else {
|
|
result.Success = false
|
|
}
|
|
for _, condition := range service.Conditions {
|
|
success := condition.evaluate(result)
|
|
if !success {
|
|
result.Success = false
|
|
}
|
|
}
|
|
result.Timestamp = time.Now()
|
|
// No need to keep the body after the service has been evaluated
|
|
result.body = nil
|
|
return result
|
|
}
|
|
|
|
func (service *Service) getIP(result *Result) {
|
|
if service.DNS != nil {
|
|
result.Hostname = strings.TrimSuffix(service.URL, ":53")
|
|
} else {
|
|
urlObject, err := url.Parse(service.URL)
|
|
if err != nil {
|
|
result.AddError(err.Error())
|
|
return
|
|
}
|
|
result.Hostname = urlObject.Hostname()
|
|
}
|
|
ips, err := net.LookupIP(result.Hostname)
|
|
if err != nil {
|
|
result.AddError(err.Error())
|
|
return
|
|
}
|
|
result.IP = ips[0].String()
|
|
}
|
|
|
|
func (service *Service) call(result *Result) {
|
|
var request *http.Request
|
|
var response *http.Response
|
|
var err error
|
|
var certificate *x509.Certificate
|
|
isServiceDNS := service.DNS != nil
|
|
isServiceTCP := strings.HasPrefix(service.URL, "tcp://")
|
|
isServiceICMP := strings.HasPrefix(service.URL, "icmp://")
|
|
isServiceStartTLS := strings.HasPrefix(service.URL, "starttls://")
|
|
isServiceHTTP := !isServiceDNS && !isServiceTCP && !isServiceICMP && !isServiceStartTLS
|
|
if isServiceHTTP {
|
|
request = service.buildHTTPRequest()
|
|
}
|
|
startTime := time.Now()
|
|
if isServiceDNS {
|
|
service.DNS.query(service.URL, result)
|
|
result.Duration = time.Since(startTime)
|
|
} else if isServiceStartTLS {
|
|
result.Connected, certificate, err = client.CanPerformStartTLS(strings.TrimPrefix(service.URL, "starttls://"), service.Insecure)
|
|
if err != nil {
|
|
result.AddError(err.Error())
|
|
return
|
|
}
|
|
result.Duration = time.Since(startTime)
|
|
result.CertificateExpiration = time.Until(certificate.NotAfter)
|
|
} else if isServiceTCP {
|
|
result.Connected = client.CanCreateTCPConnection(strings.TrimPrefix(service.URL, "tcp://"))
|
|
result.Duration = time.Since(startTime)
|
|
} else if isServiceICMP {
|
|
result.Connected, result.Duration = client.Ping(strings.TrimPrefix(service.URL, "icmp://"))
|
|
} else {
|
|
response, err = client.GetHTTPClient(service.Insecure).Do(request)
|
|
result.Duration = time.Since(startTime)
|
|
if err != nil {
|
|
result.AddError(err.Error())
|
|
return
|
|
}
|
|
defer response.Body.Close()
|
|
if response.TLS != nil && len(response.TLS.PeerCertificates) > 0 {
|
|
certificate = response.TLS.PeerCertificates[0]
|
|
result.CertificateExpiration = time.Until(certificate.NotAfter)
|
|
}
|
|
result.HTTPStatus = response.StatusCode
|
|
result.Connected = response.StatusCode > 0
|
|
// Only read the body if there's a condition that uses the BodyPlaceholder
|
|
if service.needsToReadBody() {
|
|
result.body, err = ioutil.ReadAll(response.Body)
|
|
if err != nil {
|
|
result.AddError(err.Error())
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func (service *Service) buildHTTPRequest() *http.Request {
|
|
var bodyBuffer *bytes.Buffer
|
|
if service.GraphQL {
|
|
graphQlBody := map[string]string{
|
|
"query": service.Body,
|
|
}
|
|
body, _ := json.Marshal(graphQlBody)
|
|
bodyBuffer = bytes.NewBuffer(body)
|
|
} else {
|
|
bodyBuffer = bytes.NewBuffer([]byte(service.Body))
|
|
}
|
|
request, _ := http.NewRequest(service.Method, service.URL, bodyBuffer)
|
|
for k, v := range service.Headers {
|
|
request.Header.Set(k, v)
|
|
if k == HostHeader {
|
|
request.Host = v
|
|
}
|
|
}
|
|
return request
|
|
}
|
|
|
|
// needsToReadBody checks if there's any conditions that requires the response body to be read
|
|
func (service *Service) needsToReadBody() bool {
|
|
for _, condition := range service.Conditions {
|
|
if condition.hasBodyPlaceholder() {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|