2020-06-21 12:33:43 +02:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2021-02-21 01:30:03 +01:00
|
|
|
"fmt"
|
2021-02-20 23:57:06 +01:00
|
|
|
"io"
|
2020-06-21 12:33:43 +02:00
|
|
|
"log"
|
2021-02-20 23:57:06 +01:00
|
|
|
"os"
|
2021-04-23 03:10:50 +02:00
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
2021-04-23 00:40:42 +02:00
|
|
|
"time"
|
2020-06-21 12:33:43 +02:00
|
|
|
|
2021-04-23 00:40:42 +02:00
|
|
|
"github.com/hako/durafmt"
|
2020-06-21 12:33:43 +02:00
|
|
|
"github.com/juanfont/headscale"
|
2021-02-21 01:30:03 +01:00
|
|
|
"github.com/spf13/cobra"
|
2020-06-21 12:33:43 +02:00
|
|
|
"github.com/spf13/viper"
|
2021-02-20 23:57:06 +01:00
|
|
|
"gopkg.in/yaml.v2"
|
|
|
|
"tailscale.com/tailcfg"
|
2020-06-21 12:33:43 +02:00
|
|
|
)
|
|
|
|
|
2021-02-21 01:30:03 +01:00
|
|
|
const version = "0.1"
|
|
|
|
|
|
|
|
var versionCmd = &cobra.Command{
|
|
|
|
Use: "version",
|
|
|
|
Short: "Print the version.",
|
|
|
|
Long: "The version of headscale.",
|
|
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
|
|
fmt.Println(version)
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
var headscaleCmd = &cobra.Command{
|
|
|
|
Use: "headscale",
|
|
|
|
Short: "headscale - a Tailscale control server",
|
|
|
|
Long: fmt.Sprintf(`
|
|
|
|
headscale is an open source implementation of the Tailscale control server
|
|
|
|
|
|
|
|
Juan Font Alonso <juanfontalonso@gmail.com> - 2021
|
|
|
|
https://gitlab.com/juanfont/headscale`),
|
|
|
|
}
|
|
|
|
|
|
|
|
var serveCmd = &cobra.Command{
|
|
|
|
Use: "serve",
|
|
|
|
Short: "Launches the headscale server",
|
|
|
|
Args: func(cmd *cobra.Command, args []string) error {
|
|
|
|
return nil
|
|
|
|
},
|
|
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
|
|
h, err := getHeadscaleApp()
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Error initializing: %s", err)
|
|
|
|
}
|
2021-04-23 23:16:12 +02:00
|
|
|
err = h.Serve()
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Error initializing: %s", err)
|
|
|
|
}
|
2021-02-21 01:30:03 +01:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
var registerCmd = &cobra.Command{
|
2021-02-28 00:58:09 +01:00
|
|
|
Use: "register machineID namespace",
|
2021-02-21 01:30:03 +01:00
|
|
|
Short: "Registers a machine to your network",
|
2021-02-28 00:58:09 +01:00
|
|
|
Args: func(cmd *cobra.Command, args []string) error {
|
|
|
|
if len(args) < 2 {
|
|
|
|
return fmt.Errorf("Missing parameters")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
},
|
|
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
|
|
h, err := getHeadscaleApp()
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Error initializing: %s", err)
|
|
|
|
}
|
2021-02-28 20:29:31 +01:00
|
|
|
err = h.RegisterMachine(args[0], args[1])
|
|
|
|
if err != nil {
|
|
|
|
fmt.Printf("Error: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
fmt.Println("Ook.")
|
2021-02-28 00:58:09 +01:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
var namespaceCmd = &cobra.Command{
|
|
|
|
Use: "namespace",
|
|
|
|
Short: "Manage the namespaces of Headscale",
|
|
|
|
}
|
|
|
|
|
|
|
|
var createNamespaceCmd = &cobra.Command{
|
|
|
|
Use: "create NAME",
|
|
|
|
Short: "Creates a new namespace",
|
2021-02-21 01:30:03 +01:00
|
|
|
Args: func(cmd *cobra.Command, args []string) error {
|
|
|
|
if len(args) < 1 {
|
|
|
|
return fmt.Errorf("Missing parameters")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
},
|
|
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
|
|
h, err := getHeadscaleApp()
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Error initializing: %s", err)
|
|
|
|
}
|
2021-02-28 00:58:09 +01:00
|
|
|
_, err = h.CreateNamespace(args[0])
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
fmt.Printf("Ook.\n")
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
var listNamespacesCmd = &cobra.Command{
|
|
|
|
Use: "list",
|
2021-03-14 11:38:42 +01:00
|
|
|
Short: "List all the namespaces",
|
2021-02-28 00:58:09 +01:00
|
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
|
|
h, err := getHeadscaleApp()
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Error initializing: %s", err)
|
|
|
|
}
|
|
|
|
ns, err := h.ListNamespaces()
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
fmt.Printf("ID\tName\n")
|
|
|
|
for _, n := range *ns {
|
|
|
|
fmt.Printf("%d\t%s\n", n.ID, n.Name)
|
|
|
|
}
|
2021-02-21 01:30:03 +01:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2021-03-14 11:38:42 +01:00
|
|
|
var nodeCmd = &cobra.Command{
|
|
|
|
Use: "node",
|
|
|
|
Short: "Manage the nodes of Headscale",
|
|
|
|
}
|
|
|
|
|
|
|
|
var listRoutesCmd = &cobra.Command{
|
|
|
|
Use: "list-routes NAMESPACE NODE",
|
|
|
|
Short: "List the routes exposed by this node",
|
|
|
|
Args: func(cmd *cobra.Command, args []string) error {
|
|
|
|
if len(args) < 2 {
|
|
|
|
return fmt.Errorf("Missing parameters")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
},
|
|
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
|
|
h, err := getHeadscaleApp()
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Error initializing: %s", err)
|
|
|
|
}
|
|
|
|
err = h.ListNodeRoutes(args[0], args[1])
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
var enableRouteCmd = &cobra.Command{
|
|
|
|
Use: "enable-route",
|
|
|
|
Short: "Allows exposing a route declared by this node to the rest of the nodes",
|
|
|
|
Args: func(cmd *cobra.Command, args []string) error {
|
|
|
|
if len(args) < 3 {
|
|
|
|
return fmt.Errorf("Missing parameters")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
},
|
|
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
|
|
h, err := getHeadscaleApp()
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Error initializing: %s", err)
|
|
|
|
}
|
|
|
|
err = h.EnableNodeRoute(args[0], args[1], args[2])
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2021-04-23 00:25:01 +02:00
|
|
|
var preauthkeysCmd = &cobra.Command{
|
|
|
|
Use: "preauthkey",
|
|
|
|
Short: "Handle the preauthkeys in Headscale",
|
|
|
|
}
|
|
|
|
|
|
|
|
var listPreAuthKeys = &cobra.Command{
|
|
|
|
Use: "list NAMESPACE",
|
|
|
|
Short: "List the preauthkeys for this namespace",
|
|
|
|
Args: func(cmd *cobra.Command, args []string) error {
|
|
|
|
if len(args) < 1 {
|
|
|
|
return fmt.Errorf("Missing parameters")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
},
|
|
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
|
|
h, err := getHeadscaleApp()
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Error initializing: %s", err)
|
|
|
|
}
|
|
|
|
keys, err := h.GetPreAuthKeys(args[0])
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
for _, k := range *keys {
|
|
|
|
fmt.Printf(
|
2021-04-23 00:40:42 +02:00
|
|
|
"key: %s, namespace: %s, reusable: %v, expiration: %s, created_at: %s\n",
|
2021-04-23 00:25:01 +02:00
|
|
|
k.Key,
|
|
|
|
k.Namespace.Name,
|
|
|
|
k.Reusable,
|
|
|
|
k.Expiration.Format("2006-01-02 15:04:05"),
|
|
|
|
k.CreatedAt.Format("2006-01-02 15:04:05"),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2021-04-23 00:40:42 +02:00
|
|
|
var createPreAuthKeyCmd = &cobra.Command{
|
|
|
|
Use: "create NAMESPACE",
|
|
|
|
Short: "Creates a new preauthkey in the specified namespace",
|
|
|
|
Args: func(cmd *cobra.Command, args []string) error {
|
|
|
|
if len(args) < 1 {
|
|
|
|
return fmt.Errorf("Missing parameters")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
},
|
|
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
|
|
h, err := getHeadscaleApp()
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Error initializing: %s", err)
|
|
|
|
}
|
|
|
|
reusable, _ := cmd.Flags().GetBool("reusable")
|
|
|
|
|
|
|
|
e, _ := cmd.Flags().GetString("expiration")
|
|
|
|
var expiration *time.Time
|
|
|
|
if e != "" {
|
|
|
|
duration, err := durafmt.ParseStringShort(e)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Error parsing expiration: %s", err)
|
|
|
|
}
|
|
|
|
exp := time.Now().UTC().Add(duration.Duration())
|
|
|
|
expiration = &exp
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err = h.CreatePreAuthKey(args[0], reusable, expiration)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
fmt.Printf("Ook.\n")
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2020-06-21 12:33:43 +02:00
|
|
|
func main() {
|
|
|
|
viper.SetConfigName("config")
|
2021-04-21 23:33:09 +02:00
|
|
|
viper.AddConfigPath("/etc/headscale/")
|
|
|
|
viper.AddConfigPath("$HOME/.headscale")
|
2020-06-21 12:33:43 +02:00
|
|
|
viper.AddConfigPath(".")
|
|
|
|
viper.AutomaticEnv()
|
2021-04-24 04:54:15 +02:00
|
|
|
|
|
|
|
viper.SetDefault("tls_letsencrypt_cache_dir", "/var/www/.cache")
|
|
|
|
viper.SetDefault("tls_letsencrypt_challenge_type", "HTTP-01")
|
|
|
|
|
2020-06-21 12:33:43 +02:00
|
|
|
err := viper.ReadInConfig()
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Fatal error config file: %s \n", err)
|
|
|
|
}
|
|
|
|
|
2021-04-24 04:54:15 +02:00
|
|
|
if (viper.GetString("tls_letsencrypt_hostname") != "") && ((viper.GetString("tls_cert_path") != "") || (viper.GetString("tls_key_path") != "")) {
|
|
|
|
log.Fatalf("Fatal config error: set either tls_letsencrypt_hostname or tls_cert_path/tls_key_path, not both")
|
|
|
|
}
|
|
|
|
|
|
|
|
if (viper.GetString("tls_letsencrypt_hostname") != "") && (viper.GetString("tls_letsencrypt_challenge_type") == "TLS-ALPN-01") && (!strings.HasSuffix(viper.GetString("listen_addr"), ":443")) {
|
|
|
|
log.Fatalf("Fatal config error: when using tls_letsencrypt_hostname with TLS-ALPN-01 as challenge type, listen_addr must end in :443")
|
|
|
|
}
|
|
|
|
|
|
|
|
if (viper.GetString("tls_letsencrypt_challenge_type") != "HTTP-01") && (viper.GetString("tls_letsencrypt_challenge_type") != "TLS-ALPN-01") {
|
|
|
|
log.Fatalf("Fatal config error: the only supported values for tls_letsencrypt_challenge_type are HTTP-01 and TLS-ALPN-01")
|
|
|
|
}
|
|
|
|
|
|
|
|
if !strings.HasPrefix(viper.GetString("server_url"), "http://") && !strings.HasPrefix(viper.GetString("server_url"), "https://") {
|
|
|
|
log.Fatalf("Fatal config error: server_url must start with https:// or http://")
|
|
|
|
}
|
|
|
|
|
2021-02-21 01:30:03 +01:00
|
|
|
headscaleCmd.AddCommand(versionCmd)
|
|
|
|
headscaleCmd.AddCommand(serveCmd)
|
|
|
|
headscaleCmd.AddCommand(registerCmd)
|
2021-04-23 00:25:01 +02:00
|
|
|
headscaleCmd.AddCommand(preauthkeysCmd)
|
2021-02-28 00:58:09 +01:00
|
|
|
headscaleCmd.AddCommand(namespaceCmd)
|
2021-04-23 00:25:01 +02:00
|
|
|
headscaleCmd.AddCommand(nodeCmd)
|
|
|
|
|
2021-02-28 00:58:09 +01:00
|
|
|
namespaceCmd.AddCommand(createNamespaceCmd)
|
|
|
|
namespaceCmd.AddCommand(listNamespacesCmd)
|
2021-02-21 01:30:03 +01:00
|
|
|
|
2021-03-14 11:38:42 +01:00
|
|
|
nodeCmd.AddCommand(listRoutesCmd)
|
|
|
|
nodeCmd.AddCommand(enableRouteCmd)
|
|
|
|
|
2021-04-23 00:25:01 +02:00
|
|
|
preauthkeysCmd.AddCommand(listPreAuthKeys)
|
2021-04-23 00:40:42 +02:00
|
|
|
preauthkeysCmd.AddCommand(createPreAuthKeyCmd)
|
|
|
|
createPreAuthKeyCmd.PersistentFlags().Bool("reusable", false, "Make the preauthkey reusable")
|
|
|
|
createPreAuthKeyCmd.Flags().StringP("expiration", "e", "", "Human-readable expiration of the key (30m, 24h, 365d...)")
|
2021-04-23 00:25:01 +02:00
|
|
|
|
2021-02-21 01:30:03 +01:00
|
|
|
if err := headscaleCmd.Execute(); err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
os.Exit(-1)
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2021-04-23 03:10:50 +02:00
|
|
|
func absPath(path string) string {
|
|
|
|
// If a relative path is provided, prefix it with the the directory where
|
|
|
|
// the config file was found.
|
2021-04-23 22:42:27 +02:00
|
|
|
if (path != "") && !strings.HasPrefix(path, "/") {
|
2021-04-23 03:10:50 +02:00
|
|
|
dir, _ := filepath.Split(viper.ConfigFileUsed())
|
|
|
|
if dir != "" {
|
|
|
|
path = dir + "/" + path
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return path
|
|
|
|
}
|
|
|
|
|
2021-02-21 01:30:03 +01:00
|
|
|
func getHeadscaleApp() (*headscale.Headscale, error) {
|
2021-04-23 03:10:50 +02:00
|
|
|
derpMap, err := loadDerpMap(absPath(viper.GetString("derp_map_path")))
|
2021-02-20 23:57:06 +01:00
|
|
|
if err != nil {
|
|
|
|
log.Printf("Could not load DERP servers map file: %s", err)
|
|
|
|
}
|
|
|
|
|
2020-06-21 12:33:43 +02:00
|
|
|
cfg := headscale.Config{
|
|
|
|
ServerURL: viper.GetString("server_url"),
|
|
|
|
Addr: viper.GetString("listen_addr"),
|
2021-04-23 03:10:50 +02:00
|
|
|
PrivateKeyPath: absPath(viper.GetString("private_key_path")),
|
2021-02-20 23:57:06 +01:00
|
|
|
DerpMap: derpMap,
|
2020-06-21 12:33:43 +02:00
|
|
|
|
|
|
|
DBhost: viper.GetString("db_host"),
|
|
|
|
DBport: viper.GetInt("db_port"),
|
|
|
|
DBname: viper.GetString("db_name"),
|
|
|
|
DBuser: viper.GetString("db_user"),
|
|
|
|
DBpass: viper.GetString("db_pass"),
|
2021-04-23 22:54:35 +02:00
|
|
|
|
2021-04-24 04:54:15 +02:00
|
|
|
TLSLetsEncryptHostname: viper.GetString("tls_letsencrypt_hostname"),
|
|
|
|
TLSLetsEncryptCacheDir: absPath(viper.GetString("tls_letsencrypt_cache_dir")),
|
|
|
|
TLSLetsEncryptChallengeType: viper.GetString("tls_letsencrypt_challenge_type"),
|
|
|
|
|
2021-04-23 22:54:35 +02:00
|
|
|
TLSCertPath: absPath(viper.GetString("tls_cert_path")),
|
|
|
|
TLSKeyPath: absPath(viper.GetString("tls_key_path")),
|
2020-06-21 12:33:43 +02:00
|
|
|
}
|
2021-04-23 22:54:35 +02:00
|
|
|
|
2020-06-21 12:33:43 +02:00
|
|
|
h, err := headscale.NewHeadscale(cfg)
|
|
|
|
if err != nil {
|
2021-02-21 01:30:03 +01:00
|
|
|
return nil, err
|
2020-06-21 12:33:43 +02:00
|
|
|
}
|
2021-02-21 01:30:03 +01:00
|
|
|
return h, nil
|
2020-06-21 12:33:43 +02:00
|
|
|
}
|
2021-02-20 23:57:06 +01:00
|
|
|
|
|
|
|
func loadDerpMap(path string) (*tailcfg.DERPMap, error) {
|
|
|
|
derpFile, err := os.Open(path)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer derpFile.Close()
|
|
|
|
var derpMap tailcfg.DERPMap
|
|
|
|
b, err := io.ReadAll(derpFile)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
err = yaml.Unmarshal(b, &derpMap)
|
|
|
|
return &derpMap, err
|
|
|
|
}
|