Adding upstream version 1.34.4.
Signed-off-by: Daniel Baumann <daniel@debian.org>
This commit is contained in:
parent
e393c3af3f
commit
4978089aab
4963 changed files with 677545 additions and 0 deletions
233
plugins/outputs/health/health.go
Normal file
233
plugins/outputs/health/health.go
Normal file
|
@ -0,0 +1,233 @@
|
|||
//go:generate ../../../tools/readme_config_includer/generator
|
||||
package health
|
||||
|
||||
import (
|
||||
"context"
|
||||
"crypto/tls"
|
||||
_ "embed"
|
||||
"errors"
|
||||
"net"
|
||||
"net/http"
|
||||
"net/url"
|
||||
"sync"
|
||||
"time"
|
||||
|
||||
"github.com/influxdata/telegraf"
|
||||
"github.com/influxdata/telegraf/config"
|
||||
"github.com/influxdata/telegraf/internal"
|
||||
common_tls "github.com/influxdata/telegraf/plugins/common/tls"
|
||||
"github.com/influxdata/telegraf/plugins/outputs"
|
||||
)
|
||||
|
||||
//go:embed sample.conf
|
||||
var sampleConfig string
|
||||
|
||||
const (
|
||||
defaultServiceAddress = "tcp://:8080"
|
||||
defaultReadTimeout = 5 * time.Second
|
||||
defaultWriteTimeout = 5 * time.Second
|
||||
)
|
||||
|
||||
type Checker interface {
|
||||
// Check returns true if the metrics meet its criteria.
|
||||
Check(metrics []telegraf.Metric) bool
|
||||
}
|
||||
|
||||
type Health struct {
|
||||
ServiceAddress string `toml:"service_address"`
|
||||
ReadTimeout config.Duration `toml:"read_timeout"`
|
||||
WriteTimeout config.Duration `toml:"write_timeout"`
|
||||
BasicUsername string `toml:"basic_username"`
|
||||
BasicPassword string `toml:"basic_password"`
|
||||
common_tls.ServerConfig
|
||||
|
||||
Compares []*Compares `toml:"compares"`
|
||||
Contains []*Contains `toml:"contains"`
|
||||
Log telegraf.Logger `toml:"-"`
|
||||
checkers []Checker
|
||||
|
||||
wg sync.WaitGroup
|
||||
server *http.Server
|
||||
origin string
|
||||
network string
|
||||
address string
|
||||
tlsConf *tls.Config
|
||||
|
||||
mu sync.Mutex
|
||||
healthy bool
|
||||
}
|
||||
|
||||
func (*Health) SampleConfig() string {
|
||||
return sampleConfig
|
||||
}
|
||||
|
||||
func (h *Health) Init() error {
|
||||
u, err := url.Parse(h.ServiceAddress)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
switch u.Scheme {
|
||||
case "http", "https":
|
||||
h.network = "tcp"
|
||||
h.address = u.Host
|
||||
case "unix":
|
||||
h.network = u.Scheme
|
||||
h.address = u.Path
|
||||
case "tcp4", "tcp6", "tcp":
|
||||
h.network = u.Scheme
|
||||
h.address = u.Host
|
||||
default:
|
||||
return errors.New("service_address contains invalid scheme")
|
||||
}
|
||||
|
||||
h.tlsConf, err = h.ServerConfig.TLSConfig()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
h.checkers = make([]Checker, 0)
|
||||
for i := range h.Compares {
|
||||
h.checkers = append(h.checkers, h.Compares[i])
|
||||
}
|
||||
for i := range h.Contains {
|
||||
h.checkers = append(h.checkers, h.Contains[i])
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
// Connect starts the HTTP server.
|
||||
func (h *Health) Connect() error {
|
||||
authHandler := internal.BasicAuthHandler(h.BasicUsername, h.BasicPassword, "health", onAuthError)
|
||||
|
||||
h.server = &http.Server{
|
||||
Addr: h.ServiceAddress,
|
||||
Handler: authHandler(h),
|
||||
ReadTimeout: time.Duration(h.ReadTimeout),
|
||||
WriteTimeout: time.Duration(h.WriteTimeout),
|
||||
TLSConfig: h.tlsConf,
|
||||
}
|
||||
|
||||
listener, err := h.listen()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
h.origin = h.getOrigin(listener)
|
||||
|
||||
h.Log.Infof("Listening on %s", h.origin)
|
||||
|
||||
h.wg.Add(1)
|
||||
go func() {
|
||||
defer h.wg.Done()
|
||||
err := h.server.Serve(listener)
|
||||
if err != http.ErrServerClosed {
|
||||
h.Log.Errorf("Serve error on %s: %v", h.origin, err)
|
||||
}
|
||||
h.origin = ""
|
||||
}()
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func onAuthError(_ http.ResponseWriter) {
|
||||
}
|
||||
|
||||
func (h *Health) listen() (net.Listener, error) {
|
||||
if h.tlsConf != nil {
|
||||
return tls.Listen(h.network, h.address, h.tlsConf)
|
||||
}
|
||||
return net.Listen(h.network, h.address)
|
||||
}
|
||||
|
||||
func (h *Health) ServeHTTP(rw http.ResponseWriter, _ *http.Request) {
|
||||
var code = http.StatusOK
|
||||
if !h.isHealthy() {
|
||||
code = http.StatusServiceUnavailable
|
||||
}
|
||||
|
||||
rw.Header().Set("Server", internal.ProductToken())
|
||||
http.Error(rw, http.StatusText(code), code)
|
||||
}
|
||||
|
||||
// Write runs all checks over the metric batch and adjust health state.
|
||||
func (h *Health) Write(metrics []telegraf.Metric) error {
|
||||
healthy := true
|
||||
for _, checker := range h.checkers {
|
||||
success := checker.Check(metrics)
|
||||
if !success {
|
||||
healthy = false
|
||||
}
|
||||
}
|
||||
|
||||
h.setHealthy(healthy)
|
||||
return nil
|
||||
}
|
||||
|
||||
// Close shuts down the HTTP server.
|
||||
func (h *Health) Close() error {
|
||||
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
|
||||
defer cancel()
|
||||
|
||||
err := h.server.Shutdown(ctx)
|
||||
h.wg.Wait()
|
||||
return err
|
||||
}
|
||||
|
||||
// Origin returns the URL of the HTTP server.
|
||||
func (h *Health) Origin() string {
|
||||
return h.origin
|
||||
}
|
||||
|
||||
func (h *Health) getOrigin(listener net.Listener) string {
|
||||
scheme := "http"
|
||||
if h.tlsConf != nil {
|
||||
scheme = "https"
|
||||
}
|
||||
if h.network == "unix" {
|
||||
scheme = "unix"
|
||||
}
|
||||
|
||||
switch h.network {
|
||||
case "unix":
|
||||
origin := &url.URL{
|
||||
Scheme: scheme,
|
||||
Path: listener.Addr().String(),
|
||||
}
|
||||
return origin.String()
|
||||
default:
|
||||
origin := &url.URL{
|
||||
Scheme: scheme,
|
||||
Host: listener.Addr().String(),
|
||||
}
|
||||
return origin.String()
|
||||
}
|
||||
}
|
||||
|
||||
func (h *Health) setHealthy(healthy bool) {
|
||||
h.mu.Lock()
|
||||
defer h.mu.Unlock()
|
||||
h.healthy = healthy
|
||||
}
|
||||
|
||||
func (h *Health) isHealthy() bool {
|
||||
h.mu.Lock()
|
||||
defer h.mu.Unlock()
|
||||
return h.healthy
|
||||
}
|
||||
|
||||
func NewHealth() *Health {
|
||||
return &Health{
|
||||
ServiceAddress: defaultServiceAddress,
|
||||
ReadTimeout: config.Duration(defaultReadTimeout),
|
||||
WriteTimeout: config.Duration(defaultWriteTimeout),
|
||||
healthy: true,
|
||||
}
|
||||
}
|
||||
|
||||
func init() {
|
||||
outputs.Add("health", func() telegraf.Output {
|
||||
return NewHealth()
|
||||
})
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue