2023-01-27 13:37:20 -08:00
|
|
|
// Copyright (c) Tailscale Inc & AUTHORS
|
|
|
|
// SPDX-License-Identifier: BSD-3-Clause
|
2022-03-17 20:00:54 -07:00
|
|
|
|
|
|
|
package prober
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
)
|
|
|
|
|
|
|
|
const maxHTTPBody = 4 << 20 // MiB
|
|
|
|
|
2024-03-27 15:13:34 +00:00
|
|
|
// HTTP returns a ProbeClass that healthchecks an HTTP URL.
|
2022-03-17 20:00:54 -07:00
|
|
|
//
|
2024-03-27 15:13:34 +00:00
|
|
|
// The probe function sends a GET request for url, expects an HTTP 200
|
2022-03-17 20:00:54 -07:00
|
|
|
// response, and verifies that want is present in the response
|
2022-10-12 18:41:38 +01:00
|
|
|
// body.
|
2024-03-27 15:13:34 +00:00
|
|
|
func HTTP(url, wantText string) ProbeClass {
|
|
|
|
return ProbeClass{
|
|
|
|
Probe: func(ctx context.Context) error {
|
|
|
|
return probeHTTP(ctx, url, []byte(wantText))
|
|
|
|
},
|
|
|
|
Class: "http",
|
2022-03-17 20:00:54 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func probeHTTP(ctx context.Context, url string, want []byte) error {
|
|
|
|
req, err := http.NewRequestWithContext(ctx, "GET", url, nil)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("constructing request: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get a completely new transport each time, so we don't reuse a
|
|
|
|
// past connection.
|
|
|
|
tr := http.DefaultTransport.(*http.Transport).Clone()
|
|
|
|
defer tr.CloseIdleConnections()
|
|
|
|
c := &http.Client{
|
|
|
|
Transport: tr,
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, err := c.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("fetching %q: %w", url, err)
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
if resp.StatusCode != 200 {
|
|
|
|
return fmt.Errorf("fetching %q: status code %d, want 200", url, resp.StatusCode)
|
|
|
|
}
|
|
|
|
|
2022-03-26 21:21:13 -07:00
|
|
|
bs, err := io.ReadAll(io.LimitReader(resp.Body, maxHTTPBody))
|
2022-03-17 20:00:54 -07:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("reading body of %q: %w", url, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if !bytes.Contains(bs, want) {
|
2023-12-12 19:14:25 +00:00
|
|
|
// Log response body, but truncate it if it's too large; the limit
|
|
|
|
// has been chosen arbitrarily.
|
|
|
|
if maxlen := 300; len(bs) > maxlen {
|
|
|
|
bs = bs[:maxlen]
|
|
|
|
}
|
|
|
|
return fmt.Errorf("body of %q does not contain %q (got: %q)", url, want, string(bs))
|
2022-03-17 20:00:54 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|