mirror of
https://github.com/caddyserver/caddy.git
synced 2025-01-06 22:40:31 -05:00
reverseproxy: Correct alternate port for active health checks (#3693)
* reverseproxy: construct active health-check transport from scratch (Fixes #3691) * reverseproxy: do upstream health-check on the correct alternative port * reverseproxy: add integration test for health-check on alternative port * reverseproxy: put back the custom transport for health-check http client * reverseproxy: cleanup health-check integration test * reverseproxy: fix health-check of unix socket upstreams * reverseproxy: skip unix socket tests on Windows * tabs > spaces Co-authored-by: Francis Lavoie <lavofr@gmail.com> * make the linter (and @francislavoie) happy Co-authored-by: Francis Lavoie <lavofr@gmail.com> * One more lint fix Co-authored-by: Francis Lavoie <lavofr@gmail.com> Co-authored-by: Francis Lavoie <lavofr@gmail.com>
This commit is contained in:
parent
e3324aa6de
commit
bc453fa6ae
5 changed files with 132 additions and 21 deletions
97
caddytest/integration/reverseproxy_test.go
Normal file
97
caddytest/integration/reverseproxy_test.go
Normal file
|
@ -0,0 +1,97 @@
|
||||||
|
package integration
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"io/ioutil"
|
||||||
|
"net"
|
||||||
|
"net/http"
|
||||||
|
"os"
|
||||||
|
"runtime"
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/caddyserver/caddy/v2/caddytest"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestReverseProxyHealthCheck(t *testing.T) {
|
||||||
|
tester := caddytest.NewTester(t)
|
||||||
|
tester.InitServer(`
|
||||||
|
{
|
||||||
|
http_port 9080
|
||||||
|
https_port 9443
|
||||||
|
}
|
||||||
|
http://localhost:2020 {
|
||||||
|
respond "Hello, World!"
|
||||||
|
}
|
||||||
|
http://localhost:2021 {
|
||||||
|
respond "ok"
|
||||||
|
}
|
||||||
|
http://localhost:9080 {
|
||||||
|
reverse_proxy {
|
||||||
|
to localhost:2020
|
||||||
|
|
||||||
|
health_path /health
|
||||||
|
health_port 2021
|
||||||
|
health_interval 2s
|
||||||
|
health_timeout 5s
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`, "caddyfile")
|
||||||
|
|
||||||
|
tester.AssertGetResponse("http://localhost:9080/", 200, "Hello, World!")
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestReverseProxyHealthCheckUnixSocket(t *testing.T) {
|
||||||
|
if runtime.GOOS == "windows" {
|
||||||
|
t.SkipNow()
|
||||||
|
}
|
||||||
|
tester := caddytest.NewTester(t)
|
||||||
|
f, err := ioutil.TempFile("", "*.sock")
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("failed to create TempFile: %s", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// a hack to get a file name within a valid path to use as socket
|
||||||
|
socketName := f.Name()
|
||||||
|
os.Remove(f.Name())
|
||||||
|
|
||||||
|
server := http.Server{
|
||||||
|
Handler: http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) {
|
||||||
|
if strings.HasPrefix(req.URL.Path, "/health") {
|
||||||
|
w.Write([]byte("ok"))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
w.Write([]byte("Hello, World!"))
|
||||||
|
}),
|
||||||
|
}
|
||||||
|
|
||||||
|
unixListener, err := net.Listen("unix", socketName)
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("failed to listen on the socket: %s", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
go server.Serve(unixListener)
|
||||||
|
t.Cleanup(func() {
|
||||||
|
server.Close()
|
||||||
|
})
|
||||||
|
runtime.Gosched() // Allow other goroutines to run
|
||||||
|
|
||||||
|
tester.InitServer(fmt.Sprintf(`
|
||||||
|
{
|
||||||
|
http_port 9080
|
||||||
|
https_port 9443
|
||||||
|
}
|
||||||
|
http://localhost:9080 {
|
||||||
|
reverse_proxy {
|
||||||
|
to unix/%s
|
||||||
|
|
||||||
|
health_path /health
|
||||||
|
health_port 2021
|
||||||
|
health_interval 2s
|
||||||
|
health_timeout 5s
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`, socketName), "caddyfile")
|
||||||
|
|
||||||
|
tester.AssertGetResponse("http://localhost:9080/", 200, "Hello, World!")
|
||||||
|
}
|
|
@ -153,32 +153,27 @@ func (h *Handler) doActiveHealthCheckForAllHosts() {
|
||||||
log.Printf("[PANIC] active health check: %v\n%s", err, debug.Stack())
|
log.Printf("[PANIC] active health check: %v\n%s", err, debug.Stack())
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
networkAddr := upstream.Dial
|
|
||||||
addr, err := caddy.ParseNetworkAddress(networkAddr)
|
portStr := strconv.Itoa(upstream.activeHealthCheckPort)
|
||||||
if err != nil {
|
hostAddr := net.JoinHostPort(upstream.networkAddress.Host, portStr)
|
||||||
h.HealthChecks.Active.logger.Error("bad network address",
|
if upstream.networkAddress.IsUnixNetwork() {
|
||||||
zap.String("address", networkAddr),
|
|
||||||
zap.Error(err),
|
|
||||||
)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if addr.PortRangeSize() != 1 {
|
|
||||||
h.HealthChecks.Active.logger.Error("multiple addresses (upstream must map to only one address)",
|
|
||||||
zap.String("address", networkAddr),
|
|
||||||
)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
hostAddr := addr.JoinHostPort(0)
|
|
||||||
if addr.IsUnixNetwork() {
|
|
||||||
// this will be used as the Host portion of a http.Request URL, and
|
// this will be used as the Host portion of a http.Request URL, and
|
||||||
// paths to socket files would produce an error when creating URL,
|
// paths to socket files would produce an error when creating URL,
|
||||||
// so use a fake Host value instead; unix sockets are usually local
|
// so use a fake Host value instead; unix sockets are usually local
|
||||||
hostAddr = "localhost"
|
hostAddr = "localhost"
|
||||||
}
|
}
|
||||||
err = h.doActiveHealthCheck(DialInfo{Network: addr.Network, Address: hostAddr}, hostAddr, upstream.Host)
|
|
||||||
|
dialInfo := DialInfo{
|
||||||
|
Upstream: upstream,
|
||||||
|
Network: upstream.networkAddress.Network,
|
||||||
|
Host: upstream.networkAddress.Host,
|
||||||
|
Port: portStr,
|
||||||
|
Address: hostAddr,
|
||||||
|
}
|
||||||
|
err := h.doActiveHealthCheck(dialInfo, hostAddr, upstream.Host)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
h.HealthChecks.Active.logger.Error("active health check failed",
|
h.HealthChecks.Active.logger.Error("active health check failed",
|
||||||
zap.String("address", networkAddr),
|
zap.String("address", hostAddr),
|
||||||
zap.Error(err),
|
zap.Error(err),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
|
@ -92,8 +92,10 @@ type Upstream struct {
|
||||||
// HeaderAffinity string
|
// HeaderAffinity string
|
||||||
// IPAffinity string
|
// IPAffinity string
|
||||||
|
|
||||||
healthCheckPolicy *PassiveHealthChecks
|
networkAddress caddy.NetworkAddress
|
||||||
cb CircuitBreaker
|
activeHealthCheckPort int
|
||||||
|
healthCheckPolicy *PassiveHealthChecks
|
||||||
|
cb CircuitBreaker
|
||||||
}
|
}
|
||||||
|
|
||||||
func (u Upstream) String() string {
|
func (u Upstream) String() string {
|
||||||
|
|
|
@ -182,6 +182,9 @@ func (h *HTTPTransport) NewTransport(ctx caddy.Context) (*http.Transport, error)
|
||||||
if dialInfo, ok := GetDialInfo(ctx); ok {
|
if dialInfo, ok := GetDialInfo(ctx); ok {
|
||||||
network = dialInfo.Network
|
network = dialInfo.Network
|
||||||
address = dialInfo.Address
|
address = dialInfo.Address
|
||||||
|
if dialInfo.Upstream.networkAddress.IsUnixNetwork() {
|
||||||
|
address = dialInfo.Host
|
||||||
|
}
|
||||||
}
|
}
|
||||||
conn, err := dialer.DialContext(ctx, network, address)
|
conn, err := dialer.DialContext(ctx, network, address)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|
|
@ -208,9 +208,13 @@ func (h *Handler) Provision(ctx caddy.Context) error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
if addr.PortRangeSize() != 1 {
|
if addr.PortRangeSize() != 1 {
|
||||||
return fmt.Errorf("multiple addresses (upstream must map to only one address): %v", addr)
|
return fmt.Errorf("multiple addresses (upstream must map to only one address): %v", addr)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
upstream.networkAddress = addr
|
||||||
|
|
||||||
// create or get the host representation for this upstream
|
// create or get the host representation for this upstream
|
||||||
var host Host = new(upstreamHost)
|
var host Host = new(upstreamHost)
|
||||||
existingHost, loaded := hosts.LoadOrStore(upstream.String(), host)
|
existingHost, loaded := hosts.LoadOrStore(upstream.String(), host)
|
||||||
|
@ -267,6 +271,16 @@ func (h *Handler) Provision(ctx caddy.Context) error {
|
||||||
Transport: h.Transport,
|
Transport: h.Transport,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for _, upstream := range h.Upstreams {
|
||||||
|
// if there's an alternative port for health-check provided in the config,
|
||||||
|
// then use it, otherwise use the port of upstream.
|
||||||
|
if h.HealthChecks.Active.Port != 0 {
|
||||||
|
upstream.activeHealthCheckPort = h.HealthChecks.Active.Port
|
||||||
|
} else {
|
||||||
|
upstream.activeHealthCheckPort = int(upstream.networkAddress.StartPort)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if h.HealthChecks.Active.Interval == 0 {
|
if h.HealthChecks.Active.Interval == 0 {
|
||||||
h.HealthChecks.Active.Interval = caddy.Duration(30 * time.Second)
|
h.HealthChecks.Active.Interval = caddy.Duration(30 * time.Second)
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue