2022-12-10 20:40:23 +08:00
|
|
|
package health
|
|
|
|
|
|
|
|
import (
|
2022-12-11 00:41:50 +08:00
|
|
|
"fmt"
|
2022-12-10 20:40:23 +08:00
|
|
|
"net/http"
|
2022-12-11 00:41:50 +08:00
|
|
|
"strings"
|
2022-12-10 20:40:23 +08:00
|
|
|
"sync"
|
|
|
|
|
|
|
|
"github.com/zeromicro/go-zero/core/syncx"
|
|
|
|
)
|
|
|
|
|
2023-01-29 18:01:23 +08:00
|
|
|
// defaultHealthManager is global comboHealthManager.
|
2022-12-10 20:40:23 +08:00
|
|
|
var defaultHealthManager = newComboHealthManager()
|
|
|
|
|
|
|
|
type (
|
2024-04-03 22:55:52 +08:00
|
|
|
// Probe represents readiness status of a given component.
|
2022-12-10 20:40:23 +08:00
|
|
|
Probe interface {
|
|
|
|
// MarkReady sets a ready state for the endpoint handlers.
|
|
|
|
MarkReady()
|
|
|
|
// MarkNotReady sets a not ready state for the endpoint handlers.
|
|
|
|
MarkNotReady()
|
|
|
|
// IsReady return inner state for the component.
|
|
|
|
IsReady() bool
|
|
|
|
// Name return probe name identifier
|
|
|
|
Name() string
|
|
|
|
}
|
|
|
|
|
|
|
|
// healthManager manage app healthy.
|
|
|
|
healthManager struct {
|
|
|
|
ready syncx.AtomicBool
|
|
|
|
name string
|
|
|
|
}
|
|
|
|
|
|
|
|
// comboHealthManager folds given probes into one, reflects their statuses in a thread-safe way.
|
|
|
|
comboHealthManager struct {
|
|
|
|
mu sync.Mutex
|
|
|
|
probes []Probe
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
// AddProbe add components probe to global comboHealthManager.
|
|
|
|
func AddProbe(probe Probe) {
|
|
|
|
defaultHealthManager.addProbe(probe)
|
|
|
|
}
|
|
|
|
|
2022-12-11 00:41:50 +08:00
|
|
|
// CreateHttpHandler create health http handler base on given probe.
|
2024-04-03 22:55:52 +08:00
|
|
|
func CreateHttpHandler(healthResponse string) http.HandlerFunc {
|
2022-12-11 00:41:50 +08:00
|
|
|
return func(w http.ResponseWriter, _ *http.Request) {
|
|
|
|
if defaultHealthManager.IsReady() {
|
2024-04-03 22:55:52 +08:00
|
|
|
_, _ = w.Write([]byte(healthResponse))
|
2022-12-11 00:41:50 +08:00
|
|
|
} else {
|
|
|
|
http.Error(w, "Service Unavailable\n"+defaultHealthManager.verboseInfo(),
|
|
|
|
http.StatusServiceUnavailable)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-10 20:40:23 +08:00
|
|
|
// NewHealthManager returns a new healthManager.
|
|
|
|
func NewHealthManager(name string) Probe {
|
|
|
|
return &healthManager{
|
|
|
|
name: name,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarkReady sets a ready state for the endpoint handlers.
|
|
|
|
func (h *healthManager) MarkReady() {
|
|
|
|
h.ready.Set(true)
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarkNotReady sets a not ready state for the endpoint handlers.
|
|
|
|
func (h *healthManager) MarkNotReady() {
|
|
|
|
h.ready.Set(false)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsReady return inner state for the component.
|
|
|
|
func (h *healthManager) IsReady() bool {
|
|
|
|
return h.ready.True()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Name return probe name identifier
|
|
|
|
func (h *healthManager) Name() string {
|
|
|
|
return h.name
|
|
|
|
}
|
|
|
|
|
|
|
|
func newComboHealthManager() *comboHealthManager {
|
|
|
|
return &comboHealthManager{}
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarkReady sets components status to ready.
|
|
|
|
func (p *comboHealthManager) MarkReady() {
|
|
|
|
p.mu.Lock()
|
|
|
|
defer p.mu.Unlock()
|
|
|
|
|
|
|
|
for _, probe := range p.probes {
|
|
|
|
probe.MarkReady()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarkNotReady sets components status to not ready with given error as a cause.
|
|
|
|
func (p *comboHealthManager) MarkNotReady() {
|
|
|
|
p.mu.Lock()
|
|
|
|
defer p.mu.Unlock()
|
|
|
|
|
|
|
|
for _, probe := range p.probes {
|
|
|
|
probe.MarkNotReady()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsReady return composed status of all components.
|
|
|
|
func (p *comboHealthManager) IsReady() bool {
|
|
|
|
p.mu.Lock()
|
|
|
|
defer p.mu.Unlock()
|
|
|
|
|
|
|
|
for _, probe := range p.probes {
|
|
|
|
if !probe.IsReady() {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
2022-12-11 00:41:50 +08:00
|
|
|
|
2022-12-10 20:40:23 +08:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *comboHealthManager) verboseInfo() string {
|
|
|
|
p.mu.Lock()
|
|
|
|
defer p.mu.Unlock()
|
|
|
|
|
2022-12-11 00:41:50 +08:00
|
|
|
var info strings.Builder
|
2022-12-10 20:40:23 +08:00
|
|
|
for _, probe := range p.probes {
|
|
|
|
if probe.IsReady() {
|
2022-12-11 00:41:50 +08:00
|
|
|
info.WriteString(fmt.Sprintf("%s is ready\n", probe.Name()))
|
2022-12-10 20:40:23 +08:00
|
|
|
} else {
|
2022-12-11 00:41:50 +08:00
|
|
|
info.WriteString(fmt.Sprintf("%s is not ready\n", probe.Name()))
|
2022-12-10 20:40:23 +08:00
|
|
|
}
|
|
|
|
}
|
2022-12-11 00:41:50 +08:00
|
|
|
|
|
|
|
return info.String()
|
2022-12-10 20:40:23 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// addProbe add components probe to comboHealthManager.
|
|
|
|
func (p *comboHealthManager) addProbe(probe Probe) {
|
|
|
|
p.mu.Lock()
|
|
|
|
defer p.mu.Unlock()
|
|
|
|
|
|
|
|
p.probes = append(p.probes, probe)
|
|
|
|
}
|