Files
gluetun/internal/server/vpn.go
2024-10-11 19:27:29 +00:00

132 lines
3.2 KiB
Go

package server
import (
"context"
"encoding/json"
"net/http"
"strings"
"github.com/qdm12/gluetun/internal/configuration/settings"
)
func newVPNHandler(ctx context.Context, looper VPNLooper,
storage Storage, ipv6Supported bool, w warner,
) http.Handler {
return &vpnHandler{
ctx: ctx,
looper: looper,
storage: storage,
ipv6Supported: ipv6Supported,
warner: w,
}
}
type vpnHandler struct {
ctx context.Context //nolint:containedctx
looper VPNLooper
storage Storage
ipv6Supported bool
warner warner
}
func (h *vpnHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
r.RequestURI = strings.TrimPrefix(r.RequestURI, "/vpn")
switch r.RequestURI {
case "/status":
switch r.Method {
case http.MethodGet:
h.getStatus(w)
case http.MethodPut:
h.setStatus(w, r)
default:
errMethodNotSupported(w, r.Method)
}
case "/settings":
switch r.Method {
case http.MethodGet:
h.getSettings(w)
case http.MethodPut:
h.patchSettings(w, r)
default:
errMethodNotSupported(w, r.Method)
}
default:
errRouteNotSupported(w, r.RequestURI)
}
}
func (h *vpnHandler) getStatus(w http.ResponseWriter) {
status := h.looper.GetStatus()
encoder := json.NewEncoder(w)
data := statusWrapper{Status: string(status)}
if err := encoder.Encode(data); err != nil {
h.warner.Warn(err.Error())
w.WriteHeader(http.StatusInternalServerError)
return
}
}
func (h *vpnHandler) setStatus(w http.ResponseWriter, r *http.Request) {
decoder := json.NewDecoder(r.Body)
var data statusWrapper
if err := decoder.Decode(&data); err != nil {
http.Error(w, err.Error(), http.StatusBadRequest)
return
}
status, err := data.getStatus()
if err != nil {
http.Error(w, err.Error(), http.StatusBadRequest)
return
}
outcome, err := h.looper.ApplyStatus(h.ctx, status)
if err != nil {
http.Error(w, err.Error(), http.StatusBadRequest)
return
}
encoder := json.NewEncoder(w)
if err := encoder.Encode(outcomeWrapper{Outcome: outcome}); err != nil {
h.warner.Warn(err.Error())
http.Error(w, http.StatusText(http.StatusInternalServerError), http.StatusInternalServerError)
return
}
}
func (h *vpnHandler) getSettings(w http.ResponseWriter) {
settings := h.looper.GetSettings()
encoder := json.NewEncoder(w)
if err := encoder.Encode(settings); err != nil {
h.warner.Warn(err.Error())
w.WriteHeader(http.StatusInternalServerError)
return
}
}
func (h *vpnHandler) patchSettings(w http.ResponseWriter, r *http.Request) {
var overrideSettings settings.VPN
decoder := json.NewDecoder(r.Body)
err := decoder.Decode(&overrideSettings)
if err != nil {
http.Error(w, err.Error(), http.StatusBadRequest)
return
}
err = r.Body.Close()
if err != nil {
h.warner.Warn("closing body: " + err.Error())
}
updatedSettings := h.looper.GetSettings() // already copied
updatedSettings.OverrideWith(overrideSettings)
err = updatedSettings.Validate(h.storage, h.ipv6Supported, h.warner)
if err != nil {
http.Error(w, err.Error(), http.StatusBadRequest)
return
}
outcome := h.looper.SetSettings(h.ctx, updatedSettings)
_, err = w.Write([]byte(outcome))
if err != nil {
h.warner.Warn("writing response: " + err.Error())
}
}