Health server runs on 127.0.0.1:9999, fix #272
This commit is contained in:
@@ -17,6 +17,7 @@ import (
|
|||||||
"github.com/qdm12/gluetun/internal/constants"
|
"github.com/qdm12/gluetun/internal/constants"
|
||||||
"github.com/qdm12/gluetun/internal/dns"
|
"github.com/qdm12/gluetun/internal/dns"
|
||||||
"github.com/qdm12/gluetun/internal/firewall"
|
"github.com/qdm12/gluetun/internal/firewall"
|
||||||
|
"github.com/qdm12/gluetun/internal/healthcheck"
|
||||||
gluetunLogging "github.com/qdm12/gluetun/internal/logging"
|
gluetunLogging "github.com/qdm12/gluetun/internal/logging"
|
||||||
"github.com/qdm12/gluetun/internal/openvpn"
|
"github.com/qdm12/gluetun/internal/openvpn"
|
||||||
"github.com/qdm12/gluetun/internal/params"
|
"github.com/qdm12/gluetun/internal/params"
|
||||||
@@ -52,8 +53,7 @@ func _main(background context.Context, args []string) int { //nolint:gocognit,go
|
|||||||
var err error
|
var err error
|
||||||
switch args[1] {
|
switch args[1] {
|
||||||
case "healthcheck":
|
case "healthcheck":
|
||||||
client := &http.Client{Timeout: time.Second}
|
err = cli.HealthCheck(background)
|
||||||
err = cli.HealthCheck(background, client)
|
|
||||||
case "clientkey":
|
case "clientkey":
|
||||||
err = cli.ClientKey(args[2:])
|
err = cli.ClientKey(args[2:])
|
||||||
case "openvpnconfig":
|
case "openvpnconfig":
|
||||||
@@ -259,6 +259,11 @@ func _main(background context.Context, args []string) int { //nolint:gocognit,go
|
|||||||
wg.Add(1)
|
wg.Add(1)
|
||||||
go httpServer.Run(ctx, wg)
|
go httpServer.Run(ctx, wg)
|
||||||
|
|
||||||
|
healthcheckServer := healthcheck.NewServer(
|
||||||
|
constants.HealthcheckAddress, logger)
|
||||||
|
wg.Add(1)
|
||||||
|
go healthcheckServer.Run(ctx, wg)
|
||||||
|
|
||||||
// Start openvpn for the first time
|
// Start openvpn for the first time
|
||||||
openvpnLooper.Restart()
|
openvpnLooper.Restart()
|
||||||
|
|
||||||
|
|||||||
@@ -4,12 +4,12 @@ import (
|
|||||||
"context"
|
"context"
|
||||||
"flag"
|
"flag"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io/ioutil"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/qdm12/gluetun/internal/constants"
|
"github.com/qdm12/gluetun/internal/constants"
|
||||||
|
"github.com/qdm12/gluetun/internal/healthcheck"
|
||||||
"github.com/qdm12/gluetun/internal/params"
|
"github.com/qdm12/gluetun/internal/params"
|
||||||
"github.com/qdm12/gluetun/internal/provider"
|
"github.com/qdm12/gluetun/internal/provider"
|
||||||
"github.com/qdm12/gluetun/internal/settings"
|
"github.com/qdm12/gluetun/internal/settings"
|
||||||
@@ -39,25 +39,14 @@ func ClientKey(args []string) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func HealthCheck(ctx context.Context, client *http.Client) error {
|
func HealthCheck(ctx context.Context) error {
|
||||||
const url = "http://localhost:8000/health"
|
const timeout = 3 * time.Second
|
||||||
request, err := http.NewRequestWithContext(ctx, http.MethodGet, url, nil)
|
httpClient := &http.Client{Timeout: timeout}
|
||||||
if err != nil {
|
healthchecker := healthcheck.NewChecker(httpClient)
|
||||||
return err
|
ctx, cancel := context.WithTimeout(ctx, timeout)
|
||||||
}
|
defer cancel()
|
||||||
response, err := client.Do(request)
|
const url = "http://" + constants.HealthcheckAddress
|
||||||
if err != nil {
|
return healthchecker.Check(ctx, url)
|
||||||
return err
|
|
||||||
}
|
|
||||||
defer response.Body.Close()
|
|
||||||
if response.StatusCode == http.StatusOK {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
b, err := ioutil.ReadAll(response.Body)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return fmt.Errorf("HTTP status code %s with message: %s", response.Status, string(b))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func OpenvpnConfig() error {
|
func OpenvpnConfig() error {
|
||||||
|
|||||||
5
internal/constants/addresses.go
Normal file
5
internal/constants/addresses.go
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
package constants
|
||||||
|
|
||||||
|
const (
|
||||||
|
HealthcheckAddress = "127.0.0.1:9999"
|
||||||
|
)
|
||||||
42
internal/healthcheck/client.go
Normal file
42
internal/healthcheck/client.go
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
package healthcheck
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"io/ioutil"
|
||||||
|
"net/http"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Checker interface {
|
||||||
|
Check(ctx context.Context, url string) error
|
||||||
|
}
|
||||||
|
|
||||||
|
type checker struct {
|
||||||
|
httpClient *http.Client
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewChecker(httpClient *http.Client) Checker {
|
||||||
|
return &checker{
|
||||||
|
httpClient: httpClient,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *checker) Check(ctx context.Context, url string) error {
|
||||||
|
request, err := http.NewRequestWithContext(ctx, http.MethodGet, url, nil)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
response, err := h.httpClient.Do(request)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer response.Body.Close()
|
||||||
|
if response.StatusCode == http.StatusOK {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
b, err := ioutil.ReadAll(response.Body)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return fmt.Errorf("%s: %s", response.Status, string(b))
|
||||||
|
}
|
||||||
34
internal/healthcheck/handler.go
Normal file
34
internal/healthcheck/handler.go
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
package healthcheck
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net"
|
||||||
|
"net/http"
|
||||||
|
|
||||||
|
"github.com/qdm12/golibs/logging"
|
||||||
|
)
|
||||||
|
|
||||||
|
type handler struct {
|
||||||
|
logger logging.Logger
|
||||||
|
resolver *net.Resolver
|
||||||
|
}
|
||||||
|
|
||||||
|
func newHandler(logger logging.Logger, resolver *net.Resolver) http.Handler {
|
||||||
|
return &handler{
|
||||||
|
logger: logger,
|
||||||
|
resolver: resolver,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *handler) ServeHTTP(responseWriter http.ResponseWriter, request *http.Request) {
|
||||||
|
if request.Method != http.MethodGet {
|
||||||
|
http.Error(responseWriter, "method not supported for healthcheck", http.StatusBadRequest)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
err := healthCheck(request.Context(), h.resolver)
|
||||||
|
if err != nil {
|
||||||
|
h.logger.Error(err)
|
||||||
|
http.Error(responseWriter, err.Error(), http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
responseWriter.WriteHeader(http.StatusOK)
|
||||||
|
}
|
||||||
21
internal/healthcheck/health.go
Normal file
21
internal/healthcheck/health.go
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
package healthcheck
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"net"
|
||||||
|
)
|
||||||
|
|
||||||
|
func healthCheck(ctx context.Context, resolver *net.Resolver) (err error) {
|
||||||
|
// TODO use mullvad API if current provider is Mullvad
|
||||||
|
const domainToResolve = "github.com"
|
||||||
|
ips, err := resolver.LookupIP(ctx, "ip", domainToResolve)
|
||||||
|
switch {
|
||||||
|
case err != nil:
|
||||||
|
return fmt.Errorf("cannot resolve github.com: %s", err)
|
||||||
|
case len(ips) == 0:
|
||||||
|
return fmt.Errorf("resolved no IP addresses for %s", domainToResolve)
|
||||||
|
default:
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
54
internal/healthcheck/server.go
Normal file
54
internal/healthcheck/server.go
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
package healthcheck
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"errors"
|
||||||
|
"net"
|
||||||
|
"net/http"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/qdm12/golibs/logging"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Server interface {
|
||||||
|
Run(ctx context.Context, wg *sync.WaitGroup)
|
||||||
|
}
|
||||||
|
|
||||||
|
type server struct {
|
||||||
|
address string
|
||||||
|
logger logging.Logger
|
||||||
|
handler http.Handler
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewServer(address string, logger logging.Logger) Server {
|
||||||
|
return &server{
|
||||||
|
address: address,
|
||||||
|
logger: logger.WithPrefix("healthcheck: "),
|
||||||
|
handler: newHandler(logger, &net.Resolver{}),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *server) Run(ctx context.Context, wg *sync.WaitGroup) {
|
||||||
|
server := http.Server{
|
||||||
|
Addr: s.address,
|
||||||
|
Handler: s.handler,
|
||||||
|
}
|
||||||
|
go func() {
|
||||||
|
defer wg.Done()
|
||||||
|
<-ctx.Done()
|
||||||
|
s.logger.Warn("context canceled: shutting down server")
|
||||||
|
defer s.logger.Warn("server shut down")
|
||||||
|
const shutdownGraceDuration = 2 * time.Second
|
||||||
|
shutdownCtx, cancel := context.WithTimeout(context.Background(), shutdownGraceDuration)
|
||||||
|
defer cancel()
|
||||||
|
if err := server.Shutdown(shutdownCtx); err != nil {
|
||||||
|
s.logger.Error("failed shutting down: %s", err)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
s.logger.Info("listening on %s", s.address)
|
||||||
|
err := server.ListenAndServe()
|
||||||
|
if err != nil && !errors.Is(ctx.Err(), context.Canceled) {
|
||||||
|
s.logger.Error(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,27 +0,0 @@
|
|||||||
package server
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"net/http"
|
|
||||||
)
|
|
||||||
|
|
||||||
func (s *server) handleHealth(w http.ResponseWriter) {
|
|
||||||
// TODO option to disable
|
|
||||||
// TODO use mullvad API if current provider is Mullvad
|
|
||||||
ips, err := s.lookupIP("github.com")
|
|
||||||
var errorMessage string
|
|
||||||
switch {
|
|
||||||
case err != nil:
|
|
||||||
errorMessage = fmt.Sprintf("cannot resolve github.com (%s)", err)
|
|
||||||
case len(ips) == 0:
|
|
||||||
errorMessage = "resolved no IP addresses for github.com"
|
|
||||||
default: // success
|
|
||||||
w.WriteHeader(http.StatusOK)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
s.logger.Warn(errorMessage)
|
|
||||||
w.WriteHeader(http.StatusInternalServerError)
|
|
||||||
if _, err := w.Write([]byte(errorMessage)); err != nil {
|
|
||||||
s.logger.Warn(err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -64,9 +64,7 @@ func (s *server) Run(ctx context.Context, wg *sync.WaitGroup) {
|
|||||||
|
|
||||||
func (s *server) makeHandler() http.HandlerFunc {
|
func (s *server) makeHandler() http.HandlerFunc {
|
||||||
return func(w http.ResponseWriter, r *http.Request) {
|
return func(w http.ResponseWriter, r *http.Request) {
|
||||||
if s.logging && (r.Method != http.MethodGet || r.RequestURI != "/health") {
|
s.logger.Info("HTTP %s %s", r.Method, r.RequestURI)
|
||||||
s.logger.Info("HTTP %s %s", r.Method, r.RequestURI)
|
|
||||||
}
|
|
||||||
switch r.Method {
|
switch r.Method {
|
||||||
case http.MethodGet:
|
case http.MethodGet:
|
||||||
switch r.RequestURI {
|
switch r.RequestURI {
|
||||||
@@ -80,8 +78,6 @@ func (s *server) makeHandler() http.HandlerFunc {
|
|||||||
s.handleGetPortForwarded(w)
|
s.handleGetPortForwarded(w)
|
||||||
case "/openvpn/settings":
|
case "/openvpn/settings":
|
||||||
s.handleGetOpenvpnSettings(w)
|
s.handleGetOpenvpnSettings(w)
|
||||||
case "/health":
|
|
||||||
s.handleHealth(w)
|
|
||||||
case "/updater/restart":
|
case "/updater/restart":
|
||||||
s.updaterLooper.Restart()
|
s.updaterLooper.Restart()
|
||||||
w.WriteHeader(http.StatusOK)
|
w.WriteHeader(http.StatusOK)
|
||||||
|
|||||||
Reference in New Issue
Block a user