2021-04-28 16:15:45 +02:00
|
|
|
package cli
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"log"
|
2021-08-15 23:29:55 +02:00
|
|
|
"strconv"
|
2021-05-08 13:58:51 +02:00
|
|
|
"strings"
|
2021-04-28 16:15:45 +02:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/hako/durafmt"
|
2021-08-15 23:29:55 +02:00
|
|
|
"github.com/pterm/pterm"
|
2021-04-28 16:15:45 +02:00
|
|
|
"github.com/spf13/cobra"
|
|
|
|
)
|
|
|
|
|
2021-07-25 15:07:27 +02:00
|
|
|
func init() {
|
|
|
|
rootCmd.AddCommand(preauthkeysCmd)
|
|
|
|
preauthkeysCmd.PersistentFlags().StringP("namespace", "n", "", "Namespace")
|
2021-07-25 16:26:15 +02:00
|
|
|
err := preauthkeysCmd.MarkPersistentFlagRequired("namespace")
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf(err.Error())
|
|
|
|
}
|
2021-07-25 15:07:27 +02:00
|
|
|
preauthkeysCmd.AddCommand(listPreAuthKeys)
|
|
|
|
preauthkeysCmd.AddCommand(createPreAuthKeyCmd)
|
2021-08-08 00:10:30 +02:00
|
|
|
preauthkeysCmd.AddCommand(expirePreAuthKeyCmd)
|
2021-07-25 15:07:27 +02:00
|
|
|
createPreAuthKeyCmd.PersistentFlags().Bool("reusable", false, "Make the preauthkey reusable")
|
|
|
|
createPreAuthKeyCmd.PersistentFlags().Bool("ephemeral", false, "Preauthkey for ephemeral nodes")
|
2021-07-30 16:42:26 +02:00
|
|
|
createPreAuthKeyCmd.Flags().StringP("expiration", "e", "", "Human-readable expiration of the key (30m, 24h, 365d...)")
|
2021-07-25 15:07:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
var preauthkeysCmd = &cobra.Command{
|
2021-06-28 20:04:05 +02:00
|
|
|
Use: "preauthkeys",
|
2021-04-28 16:15:45 +02:00
|
|
|
Short: "Handle the preauthkeys in Headscale",
|
|
|
|
}
|
|
|
|
|
2021-07-25 15:07:27 +02:00
|
|
|
var listPreAuthKeys = &cobra.Command{
|
2021-04-30 00:23:26 +02:00
|
|
|
Use: "list",
|
2021-04-28 16:15:45 +02:00
|
|
|
Short: "List the preauthkeys for this namespace",
|
|
|
|
Run: func(cmd *cobra.Command, args []string) {
|
2021-04-30 00:23:26 +02:00
|
|
|
n, err := cmd.Flags().GetString("namespace")
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Error getting namespace: %s", err)
|
|
|
|
}
|
2021-05-08 13:58:51 +02:00
|
|
|
o, _ := cmd.Flags().GetString("output")
|
2021-04-30 00:23:26 +02:00
|
|
|
|
2021-04-28 16:15:45 +02:00
|
|
|
h, err := getHeadscaleApp()
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Error initializing: %s", err)
|
|
|
|
}
|
2021-04-30 00:23:26 +02:00
|
|
|
keys, err := h.GetPreAuthKeys(n)
|
2021-05-08 13:58:51 +02:00
|
|
|
if strings.HasPrefix(o, "json") {
|
2021-05-08 17:06:36 +02:00
|
|
|
JsonOutput(keys, err, o)
|
2021-05-08 13:58:51 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-04-28 16:15:45 +02:00
|
|
|
if err != nil {
|
2021-05-08 13:58:51 +02:00
|
|
|
fmt.Printf("Error getting the list of keys: %s\n", err)
|
2021-04-28 16:15:45 +02:00
|
|
|
return
|
|
|
|
}
|
2021-08-15 23:29:55 +02:00
|
|
|
|
2021-10-13 22:51:55 +02:00
|
|
|
d := pterm.TableData{{"ID", "Key", "Reusable", "Ephemeral", "Used", "Expiration", "Created"}}
|
2021-04-28 16:15:45 +02:00
|
|
|
for _, k := range *keys {
|
2021-04-30 00:23:26 +02:00
|
|
|
expiration := "-"
|
|
|
|
if k.Expiration != nil {
|
|
|
|
expiration = k.Expiration.Format("2006-01-02 15:04:05")
|
|
|
|
}
|
2021-07-11 13:14:25 +02:00
|
|
|
|
|
|
|
var reusable string
|
|
|
|
if k.Ephemeral {
|
|
|
|
reusable = "N/A"
|
|
|
|
} else {
|
|
|
|
reusable = fmt.Sprintf("%v", k.Reusable)
|
|
|
|
}
|
|
|
|
|
2021-08-15 23:29:55 +02:00
|
|
|
d = append(d, []string{
|
|
|
|
strconv.FormatUint(k.ID, 10),
|
2021-04-28 16:15:45 +02:00
|
|
|
k.Key,
|
2021-07-11 13:14:25 +02:00
|
|
|
reusable,
|
2021-08-15 23:29:55 +02:00
|
|
|
strconv.FormatBool(k.Ephemeral),
|
2021-10-13 22:51:55 +02:00
|
|
|
fmt.Sprintf("%v", k.Used),
|
2021-04-30 00:23:26 +02:00
|
|
|
expiration,
|
2021-04-28 16:15:45 +02:00
|
|
|
k.CreatedAt.Format("2006-01-02 15:04:05"),
|
2021-08-15 23:29:55 +02:00
|
|
|
})
|
|
|
|
|
2021-04-28 16:15:45 +02:00
|
|
|
}
|
2021-08-15 23:35:03 +02:00
|
|
|
err = pterm.DefaultTable.WithHasHeader().WithData(d).Render()
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
2021-04-28 16:15:45 +02:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2021-07-25 15:07:27 +02:00
|
|
|
var createPreAuthKeyCmd = &cobra.Command{
|
2021-04-30 00:23:26 +02:00
|
|
|
Use: "create",
|
2021-04-28 16:15:45 +02:00
|
|
|
Short: "Creates a new preauthkey in the specified namespace",
|
|
|
|
Run: func(cmd *cobra.Command, args []string) {
|
2021-04-30 00:23:26 +02:00
|
|
|
n, err := cmd.Flags().GetString("namespace")
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Error getting namespace: %s", err)
|
|
|
|
}
|
2021-05-08 13:58:51 +02:00
|
|
|
o, _ := cmd.Flags().GetString("output")
|
2021-04-30 00:23:26 +02:00
|
|
|
|
2021-04-28 16:15:45 +02:00
|
|
|
h, err := getHeadscaleApp()
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Error initializing: %s", err)
|
|
|
|
}
|
|
|
|
reusable, _ := cmd.Flags().GetBool("reusable")
|
2021-05-23 02:15:29 +02:00
|
|
|
ephemeral, _ := cmd.Flags().GetBool("ephemeral")
|
2021-04-28 16:15:45 +02:00
|
|
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2021-05-23 02:15:29 +02:00
|
|
|
k, err := h.CreatePreAuthKey(n, reusable, ephemeral, expiration)
|
2021-05-08 13:58:51 +02:00
|
|
|
if strings.HasPrefix(o, "json") {
|
2021-05-08 17:06:36 +02:00
|
|
|
JsonOutput(k, err, o)
|
2021-05-08 13:58:51 +02:00
|
|
|
return
|
|
|
|
}
|
2021-04-28 16:15:45 +02:00
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
return
|
|
|
|
}
|
2021-08-08 18:37:23 +02:00
|
|
|
fmt.Printf("%s\n", k.Key)
|
2021-04-28 16:15:45 +02:00
|
|
|
},
|
|
|
|
}
|
2021-08-08 00:10:30 +02:00
|
|
|
|
|
|
|
var expirePreAuthKeyCmd = &cobra.Command{
|
2021-10-14 23:58:15 +02:00
|
|
|
Use: "expire KEY",
|
2021-08-08 00:10:30 +02:00
|
|
|
Short: "Expire a preauthkey",
|
|
|
|
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) {
|
|
|
|
n, err := cmd.Flags().GetString("namespace")
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Error getting namespace: %s", err)
|
|
|
|
}
|
|
|
|
o, _ := cmd.Flags().GetString("output")
|
|
|
|
|
|
|
|
h, err := getHeadscaleApp()
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Error initializing: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
k, err := h.GetPreAuthKey(n, args[0])
|
|
|
|
if err != nil {
|
2021-10-14 23:54:07 +02:00
|
|
|
if strings.HasPrefix(o, "json") {
|
|
|
|
JsonOutput(k, err, o)
|
|
|
|
return
|
|
|
|
}
|
2021-08-08 00:10:30 +02:00
|
|
|
log.Fatalf("Error getting the key: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
err = h.MarkExpirePreAuthKey(k)
|
|
|
|
if strings.HasPrefix(o, "json") {
|
|
|
|
JsonOutput(k, err, o)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
fmt.Println("Expired")
|
|
|
|
},
|
|
|
|
}
|