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.
|
|
|
|
|
2024-02-08 15:33:22 +00:00
|
|
|
package core
|
2017-09-12 18:25:55 +00:00
|
|
|
|
|
|
|
import (
|
2024-07-17 23:26:35 +00:00
|
|
|
"context"
|
2017-09-12 18:25:55 +00:00
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
Change healtcheck port into address format, redo #1197 (#1423)
As discussed in the comments in PR #1197. Also add documenation
accordingly.
One thing I'm not sure about is the simple check in health.go if the
address is usable in the GET request or not. From reading
https://pkg.go.dev/net#Dial it seems that the only non-standard address
format that would work in the `net` package but not in a GET url would
likely only be `:port`, as the others listed here are actually also
valid urls:
`For TCP, UDP and IP networks, if the host is empty or a literal
unspecified IP address, as in ":80", "0.0.0.0:80" or "[::]:80" for TCP
and UDP, "", "0.0.0.0" or "::" for IP, the local system is assumed.`
One additional thing I noticed is that while `WOODPECKER_SERVER_ADDR`
and `WOODPECKER_SERVER_ADDR` use the default value format of `:PORT`,
`WOODPECKER_SERVER` actually uses `localhost:9000`. I guess it makes a
bit of sense, considering the server might not be local to the agent,
but it looks a bit inconsistent this way. I don't think it would hurt to
make the `WOODPECKER_HEALTHCHECK_ADDR` in this format too, but then it's
different from the server flags again... :-)
2022-11-19 11:06:51 +00:00
|
|
|
"strings"
|
2017-09-12 18:25:55 +00:00
|
|
|
|
2021-11-23 14:36:52 +00:00
|
|
|
"github.com/rs/zerolog/log"
|
2024-07-17 23:26:35 +00:00
|
|
|
"github.com/urfave/cli/v3"
|
2021-10-12 07:25:13 +00:00
|
|
|
|
2023-12-08 07:15:08 +00:00
|
|
|
"go.woodpecker-ci.org/woodpecker/v2/agent"
|
|
|
|
"go.woodpecker-ci.org/woodpecker/v2/version"
|
2017-09-12 18:25:55 +00:00
|
|
|
)
|
|
|
|
|
2024-05-13 20:58:21 +00:00
|
|
|
// The file implements some basic healthcheck logic based on the
|
2017-09-12 18:25:55 +00:00
|
|
|
// following specification:
|
|
|
|
// https://github.com/mozilla-services/Dockerflow
|
|
|
|
|
2024-01-09 20:35:37 +00:00
|
|
|
func initHealth() {
|
2024-06-04 06:30:54 +00:00
|
|
|
http.HandleFunc("/varz", handleStats)
|
|
|
|
http.HandleFunc("/healthz", handleHeartbeat)
|
2017-09-12 20:40:24 +00:00
|
|
|
http.HandleFunc("/version", handleVersion)
|
2017-09-12 18:25:55 +00:00
|
|
|
}
|
|
|
|
|
2023-03-18 19:35:27 +00:00
|
|
|
func handleHeartbeat(w http.ResponseWriter, _ *http.Request) {
|
2017-09-12 20:40:24 +00:00
|
|
|
if counter.Healthy() {
|
2024-03-15 17:00:25 +00:00
|
|
|
w.WriteHeader(http.StatusOK)
|
2017-09-12 20:40:24 +00:00
|
|
|
} else {
|
2024-03-15 17:00:25 +00:00
|
|
|
w.WriteHeader(http.StatusInternalServerError)
|
2017-09-12 20:40:24 +00:00
|
|
|
}
|
2017-09-12 18:25:55 +00:00
|
|
|
}
|
|
|
|
|
2023-03-18 19:35:27 +00:00
|
|
|
func handleVersion(w http.ResponseWriter, _ *http.Request) {
|
2024-03-15 17:00:25 +00:00
|
|
|
w.WriteHeader(http.StatusOK)
|
2017-09-12 18:25:55 +00:00
|
|
|
w.Header().Add("Content-Type", "text/json")
|
2024-01-09 20:35:37 +00:00
|
|
|
err := json.NewEncoder(w).Encode(versionResp{
|
2021-05-25 12:08:27 +00:00
|
|
|
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
|
|
|
})
|
2024-01-09 20:35:37 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Error().Err(err).Msg("handleVersion")
|
|
|
|
}
|
2017-09-12 18:25:55 +00:00
|
|
|
}
|
|
|
|
|
2023-03-18 19:35:27 +00:00
|
|
|
func handleStats(w http.ResponseWriter, _ *http.Request) {
|
2017-09-12 20:40:24 +00:00
|
|
|
if counter.Healthy() {
|
2024-03-15 17:00:25 +00:00
|
|
|
w.WriteHeader(http.StatusOK)
|
2017-09-12 20:40:24 +00:00
|
|
|
} else {
|
2024-03-15 17:00:25 +00:00
|
|
|
w.WriteHeader(http.StatusInternalServerError)
|
2017-09-12 20:40:24 +00:00
|
|
|
}
|
|
|
|
w.Header().Add("Content-Type", "text/json")
|
2021-11-23 14:36:52 +00:00
|
|
|
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"`
|
|
|
|
}
|
|
|
|
|
2024-05-13 20:58:21 +00:00
|
|
|
// Default statistics counter.
|
2021-09-23 14:58:12 +00:00
|
|
|
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.
|
2024-07-17 23:26:35 +00:00
|
|
|
func pinger(ctx context.Context, c *cli.Command) error {
|
Change healtcheck port into address format, redo #1197 (#1423)
As discussed in the comments in PR #1197. Also add documenation
accordingly.
One thing I'm not sure about is the simple check in health.go if the
address is usable in the GET request or not. From reading
https://pkg.go.dev/net#Dial it seems that the only non-standard address
format that would work in the `net` package but not in a GET url would
likely only be `:port`, as the others listed here are actually also
valid urls:
`For TCP, UDP and IP networks, if the host is empty or a literal
unspecified IP address, as in ":80", "0.0.0.0:80" or "[::]:80" for TCP
and UDP, "", "0.0.0.0" or "::" for IP, the local system is assumed.`
One additional thing I noticed is that while `WOODPECKER_SERVER_ADDR`
and `WOODPECKER_SERVER_ADDR` use the default value format of `:PORT`,
`WOODPECKER_SERVER` actually uses `localhost:9000`. I guess it makes a
bit of sense, considering the server might not be local to the agent,
but it looks a bit inconsistent this way. I don't think it would hurt to
make the `WOODPECKER_HEALTHCHECK_ADDR` in this format too, but then it's
different from the server flags again... :-)
2022-11-19 11:06:51 +00:00
|
|
|
healthcheckAddress := c.String("healthcheck-addr")
|
|
|
|
if strings.HasPrefix(healthcheckAddress, ":") {
|
|
|
|
// this seems sufficient according to https://pkg.go.dev/net#Dial
|
|
|
|
healthcheckAddress = "localhost" + healthcheckAddress
|
|
|
|
}
|
2024-07-13 23:06:20 +00:00
|
|
|
|
|
|
|
req, err := http.NewRequestWithContext(ctx, http.MethodGet, "http://"+healthcheckAddress+"/healthz", nil)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
resp, err := http.DefaultClient.Do(req)
|
2017-09-12 18:25:55 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
2024-03-15 17:00:25 +00:00
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
|
|
return fmt.Errorf("agent returned non-http.StatusOK status code")
|
2017-09-12 18:25:55 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|