7e71986493
Today each storagenode should have a port which is opened for the internet, and handles DRPC protocol calls. When we do a HTTP call on the DRPC endpoint, it hangs until a timeout. This patch changes the behavior: the main DRPC port of the storagenodes can accept HTTP requests and can be used to monitor the status of the node: * if returns with HTTP 200 only if the storagnode is healthy (not suspended / disqualified + online score > 0.9) * it CAN include information about the current status (per satellite). It's opt-in, you should configure it so. In this way it becomes extremely easy to monitor storagenodes with external uptime services. Note: this patch exposes some information which was not easily available before (especially the node status, and used satellites). I think it should be acceptable: * Until having more community satellites, all storagenodes are connected to the main Storj satellites. * With community satellites, it's good thing to have more transparency (easy way to check who is connected to which satellites) The implementation is based on this line: ``` http.Serve(NewPrefixedListener([]byte("GET / HT"), publicMux.Route("GET / HT")), p.public.http) ``` This line answers to the TCP requests with `GET / HT...` (GET HTTP request to the route), but puts back the removed prefix. Change-Id: I3700c7e24524850825ecdf75a4bcc3b4afcb3a74
47 lines
1.0 KiB
Go
47 lines
1.0 KiB
Go
// Copyright (C) 2022 Storj Labs, Inc.
|
|
// See LICENSE for copying information.
|
|
|
|
package healthcheck
|
|
|
|
import (
|
|
"encoding/json"
|
|
"net/http"
|
|
)
|
|
|
|
// Endpoint handles HTTP request for health endpoint.
|
|
type Endpoint struct {
|
|
service *Service
|
|
}
|
|
|
|
// NewEndpoint creates a new HTTP endpoint.
|
|
func NewEndpoint(service *Service) *Endpoint {
|
|
return &Endpoint{
|
|
service: service,
|
|
}
|
|
}
|
|
|
|
// HandleHTTP manages the HTTP conversion for the function call.
|
|
func (e *Endpoint) HandleHTTP(writer http.ResponseWriter, request *http.Request) {
|
|
health, err := e.service.GetHealth(request.Context())
|
|
if err != nil {
|
|
writer.WriteHeader(http.StatusInternalServerError)
|
|
_, _ = writer.Write([]byte(err.Error()))
|
|
return
|
|
}
|
|
|
|
out, err := json.MarshalIndent(health, "", " ")
|
|
if err != nil {
|
|
writer.WriteHeader(http.StatusInternalServerError)
|
|
_, _ = writer.Write([]byte(err.Error()))
|
|
return
|
|
}
|
|
|
|
if health.AllHealthy {
|
|
writer.WriteHeader(http.StatusOK)
|
|
} else {
|
|
writer.WriteHeader(http.StatusServiceUnavailable)
|
|
}
|
|
|
|
_, _ = writer.Write(out)
|
|
}
|