2020-06-21 06:32:08 -04:00
|
|
|
package headscale
|
|
|
|
|
|
|
|
import (
|
2021-04-23 22:54:15 -04:00
|
|
|
"errors"
|
2020-06-21 06:32:08 -04:00
|
|
|
"fmt"
|
2021-04-23 22:54:15 -04:00
|
|
|
"log"
|
|
|
|
"net/http"
|
2021-02-21 17:54:15 -05:00
|
|
|
"os"
|
2021-04-23 16:54:35 -04:00
|
|
|
"strings"
|
2021-02-23 15:07:52 -05:00
|
|
|
"sync"
|
2020-06-21 06:32:08 -04:00
|
|
|
|
|
|
|
"github.com/gin-gonic/gin"
|
2021-04-23 22:54:15 -04:00
|
|
|
"golang.org/x/crypto/acme/autocert"
|
2021-02-20 17:57:06 -05:00
|
|
|
"tailscale.com/tailcfg"
|
2021-02-20 16:43:07 -05:00
|
|
|
"tailscale.com/wgengine/wgcfg"
|
2020-06-21 06:32:08 -04:00
|
|
|
)
|
|
|
|
|
2021-02-21 16:14:38 -05:00
|
|
|
// Config contains the initial Headscale configuration
|
2020-06-21 06:32:08 -04:00
|
|
|
type Config struct {
|
|
|
|
ServerURL string
|
|
|
|
Addr string
|
|
|
|
PrivateKeyPath string
|
2021-02-20 17:57:06 -05:00
|
|
|
DerpMap *tailcfg.DERPMap
|
2020-06-21 06:32:08 -04:00
|
|
|
|
|
|
|
DBhost string
|
|
|
|
DBport int
|
|
|
|
DBname string
|
|
|
|
DBuser string
|
|
|
|
DBpass string
|
2021-04-23 16:54:35 -04:00
|
|
|
|
2021-04-23 22:54:15 -04:00
|
|
|
TLSLetsEncryptHostname string
|
|
|
|
TLSLetsEncryptCacheDir string
|
|
|
|
TLSLetsEncryptChallengeType string
|
|
|
|
|
2021-04-23 16:54:35 -04:00
|
|
|
TLSCertPath string
|
|
|
|
TLSKeyPath string
|
2020-06-21 06:32:08 -04:00
|
|
|
}
|
|
|
|
|
2021-02-21 16:14:38 -05:00
|
|
|
// Headscale represents the base app of the service
|
2020-06-21 06:32:08 -04:00
|
|
|
type Headscale struct {
|
|
|
|
cfg Config
|
|
|
|
dbString string
|
2021-05-02 14:47:36 -04:00
|
|
|
dbType string
|
|
|
|
dbDebug bool
|
2020-06-21 06:32:08 -04:00
|
|
|
publicKey *wgcfg.Key
|
|
|
|
privateKey *wgcfg.PrivateKey
|
2021-02-23 15:07:52 -05:00
|
|
|
|
|
|
|
pollMu sync.Mutex
|
|
|
|
clientsPolling map[uint64]chan []byte // this is by all means a hackity hack
|
2020-06-21 06:32:08 -04:00
|
|
|
}
|
|
|
|
|
2021-02-21 16:14:38 -05:00
|
|
|
// NewHeadscale returns the Headscale app
|
2020-06-21 06:32:08 -04:00
|
|
|
func NewHeadscale(cfg Config) (*Headscale, error) {
|
2021-02-21 17:54:15 -05:00
|
|
|
content, err := os.ReadFile(cfg.PrivateKeyPath)
|
2020-06-21 06:32:08 -04:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
privKey, err := wgcfg.ParsePrivateKey(string(content))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
pubKey := privKey.Public()
|
|
|
|
h := Headscale{
|
2021-05-02 14:47:36 -04:00
|
|
|
cfg: cfg,
|
|
|
|
dbType: "postgres",
|
2020-06-21 06:32:08 -04:00
|
|
|
dbString: fmt.Sprintf("host=%s port=%d dbname=%s user=%s password=%s sslmode=disable", cfg.DBhost,
|
|
|
|
cfg.DBport, cfg.DBname, cfg.DBuser, cfg.DBpass),
|
|
|
|
privateKey: privKey,
|
|
|
|
publicKey: &pubKey,
|
|
|
|
}
|
|
|
|
err = h.initDB()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-02-23 15:07:52 -05:00
|
|
|
h.clientsPolling = make(map[uint64]chan []byte)
|
2020-06-21 06:32:08 -04:00
|
|
|
return &h, nil
|
|
|
|
}
|
|
|
|
|
2021-04-23 22:54:15 -04:00
|
|
|
// Redirect to our TLS url
|
|
|
|
func (h *Headscale) redirect(w http.ResponseWriter, req *http.Request) {
|
|
|
|
target := h.cfg.ServerURL + req.URL.RequestURI()
|
|
|
|
http.Redirect(w, req, target, http.StatusFound)
|
|
|
|
}
|
|
|
|
|
2021-02-21 16:14:38 -05:00
|
|
|
// Serve launches a GIN server with the Headscale API
|
2020-06-21 06:32:08 -04:00
|
|
|
func (h *Headscale) Serve() error {
|
|
|
|
r := gin.Default()
|
|
|
|
r.GET("/key", h.KeyHandler)
|
|
|
|
r.GET("/register", h.RegisterWebAPI)
|
|
|
|
r.POST("/machine/:id/map", h.PollNetMapHandler)
|
|
|
|
r.POST("/machine/:id", h.RegistrationHandler)
|
2021-04-23 16:54:35 -04:00
|
|
|
var err error
|
2021-04-23 22:54:15 -04:00
|
|
|
if h.cfg.TLSLetsEncryptHostname != "" {
|
|
|
|
if !strings.HasPrefix(h.cfg.ServerURL, "https://") {
|
|
|
|
fmt.Println("WARNING: listening with TLS but ServerURL does not start with https://")
|
|
|
|
}
|
|
|
|
|
|
|
|
m := autocert.Manager{
|
|
|
|
Prompt: autocert.AcceptTOS,
|
|
|
|
HostPolicy: autocert.HostWhitelist(h.cfg.TLSLetsEncryptHostname),
|
|
|
|
Cache: autocert.DirCache(h.cfg.TLSLetsEncryptCacheDir),
|
|
|
|
}
|
|
|
|
s := &http.Server{
|
|
|
|
Addr: h.cfg.Addr,
|
|
|
|
TLSConfig: m.TLSConfig(),
|
|
|
|
Handler: r,
|
|
|
|
}
|
|
|
|
if h.cfg.TLSLetsEncryptChallengeType == "TLS-ALPN-01" {
|
|
|
|
// Configuration via autocert with TLS-ALPN-01 (https://tools.ietf.org/html/rfc8737)
|
|
|
|
// The RFC requires that the validation is done on port 443; in other words, headscale
|
|
|
|
// must be configured to run on port 443.
|
|
|
|
err = s.ListenAndServeTLS("", "")
|
|
|
|
} else if h.cfg.TLSLetsEncryptChallengeType == "HTTP-01" {
|
|
|
|
// Configuration via autocert with HTTP-01. This requires listening on
|
|
|
|
// port 80 for the certificate validation in addition to the headscale
|
|
|
|
// service, which can be configured to run on any other port.
|
|
|
|
go func() {
|
|
|
|
log.Fatal(http.ListenAndServe(":http", m.HTTPHandler(http.HandlerFunc(h.redirect))))
|
|
|
|
}()
|
2021-04-24 11:26:50 -04:00
|
|
|
err = s.ListenAndServeTLS("", "")
|
2021-04-23 22:54:15 -04:00
|
|
|
} else {
|
|
|
|
return errors.New("Unknown value for TLSLetsEncryptChallengeType")
|
|
|
|
}
|
|
|
|
} else if h.cfg.TLSCertPath == "" {
|
2021-04-23 16:54:35 -04:00
|
|
|
if !strings.HasPrefix(h.cfg.ServerURL, "http://") {
|
|
|
|
fmt.Println("WARNING: listening without TLS but ServerURL does not start with http://")
|
|
|
|
}
|
|
|
|
err = r.Run(h.cfg.Addr)
|
|
|
|
} else {
|
|
|
|
if !strings.HasPrefix(h.cfg.ServerURL, "https://") {
|
|
|
|
fmt.Println("WARNING: listening with TLS but ServerURL does not start with https://")
|
|
|
|
}
|
|
|
|
err = r.RunTLS(h.cfg.Addr, h.cfg.TLSCertPath, h.cfg.TLSKeyPath)
|
|
|
|
}
|
2020-06-21 06:32:08 -04:00
|
|
|
return err
|
|
|
|
}
|