woodpecker/cmd/agent/health.go

97 lines
2.5 KiB
Go
Raw Normal View History

2018-02-19 22:24:10 +00:00
// Copyright 2018 Drone.IO Inc.
2018-03-21 13:02:17 +00:00
//
2018-02-19 22:24:10 +00:00
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
2018-03-21 13:02:17 +00:00
//
2018-02-19 22:24:10 +00:00
// http://www.apache.org/licenses/LICENSE-2.0
2018-03-21 13:02:17 +00:00
//
2018-02-19 22:24:10 +00:00
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
2017-09-12 18:25:55 +00:00
package main
import (
"encoding/json"
"fmt"
"net/http"
"strings"
2017-09-12 18:25:55 +00:00
"github.com/rs/zerolog/log"
"github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/agent"
"github.com/woodpecker-ci/woodpecker/version"
2017-09-12 18:25:55 +00:00
)
// the file implements some basic healthcheck logic based on the
// following specification:
// https://github.com/mozilla-services/Dockerflow
func init() {
2017-09-12 20:40:24 +00:00
http.HandleFunc("/varz", handleStats)
http.HandleFunc("/healthz", handleHeartbeat)
http.HandleFunc("/version", handleVersion)
2017-09-12 18:25:55 +00:00
}
func handleHeartbeat(w http.ResponseWriter, r *http.Request) {
2017-09-12 20:40:24 +00:00
if counter.Healthy() {
w.WriteHeader(200)
} else {
w.WriteHeader(500)
}
2017-09-12 18:25:55 +00:00
}
func handleVersion(w http.ResponseWriter, r *http.Request) {
w.WriteHeader(200)
w.Header().Add("Content-Type", "text/json")
_ = json.NewEncoder(w).Encode(versionResp{
Source: "https://github.com/woodpecker-ci/woodpecker",
2019-11-12 20:49:38 +00:00
Version: version.String(),
2017-09-12 18:25:55 +00:00
})
}
2017-09-12 20:40:24 +00:00
func handleStats(w http.ResponseWriter, r *http.Request) {
if counter.Healthy() {
w.WriteHeader(200)
} else {
w.WriteHeader(500)
}
w.Header().Add("Content-Type", "text/json")
if _, err := counter.WriteTo(w); err != nil {
log.Error().Err(err).Msg("handleStats")
}
2017-09-12 20:40:24 +00:00
}
2017-09-12 18:25:55 +00:00
type versionResp struct {
Version string `json:"version"`
Source string `json:"source"`
}
2017-09-12 20:40:24 +00:00
// default statistics counter
var counter = &agent.State{
Metadata: map[string]agent.Info{},
2017-09-12 20:40:24 +00:00
}
2017-09-12 18:25:55 +00:00
// handles pinging the endpoint and returns an error if the
// agent is in an unhealthy state.
func pinger(c *cli.Context) error {
healthcheckAddress := c.String("healthcheck-addr")
if strings.HasPrefix(healthcheckAddress, ":") {
// this seems sufficient according to https://pkg.go.dev/net#Dial
healthcheckAddress = "localhost" + healthcheckAddress
}
resp, err := http.Get("http://" + healthcheckAddress + "/healthz")
2017-09-12 18:25:55 +00:00
if err != nil {
return err
}
defer resp.Body.Close()
if resp.StatusCode != 200 {
return fmt.Errorf("agent returned non-200 status code")
}
return nil
}