229 lines
5.2 KiB
Go
229 lines
5.2 KiB
Go
// Package websvc contains the AdGuard Home web service.
|
|
//
|
|
// TODO(a.garipov): Add tests.
|
|
package websvc
|
|
|
|
import (
|
|
"context"
|
|
"crypto/tls"
|
|
"fmt"
|
|
"io"
|
|
"net"
|
|
"net/http"
|
|
"net/netip"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/AdguardTeam/AdGuardHome/internal/v1/agh"
|
|
"github.com/AdguardTeam/golibs/errors"
|
|
"github.com/AdguardTeam/golibs/log"
|
|
httptreemux "github.com/dimfeld/httptreemux/v5"
|
|
)
|
|
|
|
// Config is the AdGuard Home web service configuration structure.
|
|
type Config struct {
|
|
// TLS is the optional TLS configuration. If TLS is not nil,
|
|
// SecureAddresses must not be empty.
|
|
TLS *tls.Config
|
|
|
|
// Addresses are the addresses on which to serve the plain HTTP API.
|
|
Addresses []netip.AddrPort
|
|
|
|
// SecureAddresses are the addresses on which to serve the HTTPS API. If
|
|
// SecureAddresses is not empty, TLS must not be nil.
|
|
SecureAddresses []netip.AddrPort
|
|
|
|
// Start is the time of start of AdGuard Home.
|
|
Start time.Time
|
|
|
|
// Timeout is the timeout for all server operations.
|
|
Timeout time.Duration
|
|
}
|
|
|
|
// Service is the AdGuard Home web service. A nil *Service is a valid
|
|
// [agh.Service] that does nothing.
|
|
type Service struct {
|
|
tls *tls.Config
|
|
servers []*http.Server
|
|
start time.Time
|
|
timeout time.Duration
|
|
}
|
|
|
|
// New returns a new properly initialized *Service. If c is nil, svc is a nil
|
|
// *Service that does nothing.
|
|
func New(c *Config) (svc *Service) {
|
|
if c == nil {
|
|
return nil
|
|
}
|
|
|
|
svc = &Service{
|
|
tls: c.TLS,
|
|
start: c.Start,
|
|
timeout: c.Timeout,
|
|
}
|
|
|
|
mux := newMux(svc)
|
|
|
|
for _, a := range c.Addresses {
|
|
addr := a.String()
|
|
errLog := log.StdLog("websvc: http: "+addr, log.ERROR)
|
|
svc.servers = append(svc.servers, &http.Server{
|
|
Addr: addr,
|
|
Handler: mux,
|
|
ErrorLog: errLog,
|
|
ReadTimeout: c.Timeout,
|
|
WriteTimeout: c.Timeout,
|
|
IdleTimeout: c.Timeout,
|
|
ReadHeaderTimeout: c.Timeout,
|
|
})
|
|
}
|
|
|
|
for _, a := range c.SecureAddresses {
|
|
addr := a.String()
|
|
errLog := log.StdLog("websvc: https: "+addr, log.ERROR)
|
|
svc.servers = append(svc.servers, &http.Server{
|
|
Addr: addr,
|
|
Handler: mux,
|
|
TLSConfig: c.TLS,
|
|
ErrorLog: errLog,
|
|
ReadTimeout: c.Timeout,
|
|
WriteTimeout: c.Timeout,
|
|
IdleTimeout: c.Timeout,
|
|
ReadHeaderTimeout: c.Timeout,
|
|
})
|
|
}
|
|
|
|
return svc
|
|
}
|
|
|
|
// newMux returns a new HTTP request multiplexor for the AdGuard Home web
|
|
// service.
|
|
func newMux(svc *Service) (mux *httptreemux.ContextMux) {
|
|
mux = httptreemux.NewContextMux()
|
|
|
|
routes := []struct {
|
|
handler http.HandlerFunc
|
|
method string
|
|
path string
|
|
isJSON bool
|
|
}{{
|
|
handler: svc.handleGetHealthCheck,
|
|
method: http.MethodGet,
|
|
path: PathHealthCheck,
|
|
isJSON: false,
|
|
}, {
|
|
handler: svc.handleGetV1SystemInfo,
|
|
method: http.MethodGet,
|
|
path: PathV1SystemInfo,
|
|
isJSON: true,
|
|
}}
|
|
|
|
for _, r := range routes {
|
|
var h http.HandlerFunc
|
|
if r.isJSON {
|
|
// TODO(a.garipov): Consider using httptreemux's MiddlewareFunc.
|
|
h = jsonMw(r.handler)
|
|
} else {
|
|
h = r.handler
|
|
}
|
|
|
|
mux.Handle(r.method, r.path, h)
|
|
}
|
|
|
|
return mux
|
|
}
|
|
|
|
// Addrs returns all addresses on which this server serves the HTTP API. Addrs
|
|
// must not be called until Start returns.
|
|
func (svc *Service) Addrs() (addrs []string) {
|
|
addrs = make([]string, 0, len(svc.servers))
|
|
for _, srv := range svc.servers {
|
|
addrs = append(addrs, srv.Addr)
|
|
}
|
|
|
|
return addrs
|
|
}
|
|
|
|
// handleGetHealthCheck is the handler for the GET /health-check HTTP API.
|
|
func (svc *Service) handleGetHealthCheck(w http.ResponseWriter, _ *http.Request) {
|
|
_, _ = io.WriteString(w, "OK")
|
|
}
|
|
|
|
// unit is a convenient alias for struct{}.
|
|
type unit = struct{}
|
|
|
|
// type check
|
|
var _ agh.Service = (*Service)(nil)
|
|
|
|
// Start implements the [agh.Service] interface for *Service. svc may be nil.
|
|
// After Start exits, all HTTP servers have tried to start, possibly failing and
|
|
// writing error messages to the log.
|
|
func (svc *Service) Start() (err error) {
|
|
if svc == nil {
|
|
return nil
|
|
}
|
|
|
|
srvs := svc.servers
|
|
|
|
wg := &sync.WaitGroup{}
|
|
wg.Add(len(srvs))
|
|
for _, srv := range srvs {
|
|
go serve(srv, wg)
|
|
}
|
|
|
|
wg.Wait()
|
|
|
|
return nil
|
|
}
|
|
|
|
// serve starts and runs srv and writes all errors into its log.
|
|
func serve(srv *http.Server, wg *sync.WaitGroup) {
|
|
addr := srv.Addr
|
|
defer log.OnPanic(addr)
|
|
|
|
var l net.Listener
|
|
var err error
|
|
if srv.TLSConfig == nil {
|
|
l, err = net.Listen("tcp", addr)
|
|
} else {
|
|
l, err = tls.Listen("tcp", addr, srv.TLSConfig)
|
|
}
|
|
if err != nil {
|
|
srv.ErrorLog.Printf("starting srv %s: binding: %s", addr, err)
|
|
}
|
|
|
|
// Update the server's address in case the address had the port zero, which
|
|
// would mean that a random available port was automatically chosen.
|
|
srv.Addr = l.Addr().String()
|
|
|
|
log.Info("websvc: starting srv http://%s", srv.Addr)
|
|
wg.Done()
|
|
|
|
err = srv.Serve(l)
|
|
if err != nil && !errors.Is(err, http.ErrServerClosed) {
|
|
srv.ErrorLog.Printf("starting srv %s: %s", addr, err)
|
|
}
|
|
}
|
|
|
|
// Shutdown implements the [agh.Service] interface for *Service. svc may be
|
|
// nil.
|
|
func (svc *Service) Shutdown(ctx context.Context) (err error) {
|
|
if svc == nil {
|
|
return nil
|
|
}
|
|
|
|
var errs []error
|
|
for _, srv := range svc.servers {
|
|
serr := srv.Shutdown(ctx)
|
|
if serr != nil {
|
|
errs = append(errs, fmt.Errorf("shutting down srv %s: %w", srv.Addr, serr))
|
|
}
|
|
}
|
|
|
|
if len(errs) > 0 {
|
|
return errors.List("shutting down")
|
|
}
|
|
|
|
return nil
|
|
}
|