mirror of
https://github.com/juanfont/headscale.git
synced 2025-01-18 00:06:09 +01:00
14e29a7bee
This is step one in detaching the Database layer from Headscale (h). The ultimate goal is to have all function that does database operations in its own package, and keep the business logic and writing separate. Signed-off-by: Kristoffer Dalby <kristoffer@tailscale.com>
202 lines
4.3 KiB
Go
202 lines
4.3 KiB
Go
package cli
|
|
|
|
import (
|
|
"fmt"
|
|
"strconv"
|
|
"time"
|
|
|
|
v1 "github.com/juanfont/headscale/gen/go/headscale/v1"
|
|
"github.com/juanfont/headscale/hscontrol/util"
|
|
"github.com/prometheus/common/model"
|
|
"github.com/pterm/pterm"
|
|
"github.com/rs/zerolog/log"
|
|
"github.com/spf13/cobra"
|
|
"google.golang.org/protobuf/types/known/timestamppb"
|
|
)
|
|
|
|
const (
|
|
// 90 days.
|
|
DefaultAPIKeyExpiry = "90d"
|
|
)
|
|
|
|
func init() {
|
|
rootCmd.AddCommand(apiKeysCmd)
|
|
apiKeysCmd.AddCommand(listAPIKeys)
|
|
|
|
createAPIKeyCmd.Flags().
|
|
StringP("expiration", "e", DefaultAPIKeyExpiry, "Human-readable expiration of the key (e.g. 30m, 24h)")
|
|
|
|
apiKeysCmd.AddCommand(createAPIKeyCmd)
|
|
|
|
expireAPIKeyCmd.Flags().StringP("prefix", "p", "", "ApiKey prefix")
|
|
err := expireAPIKeyCmd.MarkFlagRequired("prefix")
|
|
if err != nil {
|
|
log.Fatal().Err(err).Msg("")
|
|
}
|
|
apiKeysCmd.AddCommand(expireAPIKeyCmd)
|
|
}
|
|
|
|
var apiKeysCmd = &cobra.Command{
|
|
Use: "apikeys",
|
|
Short: "Handle the Api keys in Headscale",
|
|
Aliases: []string{"apikey", "api"},
|
|
}
|
|
|
|
var listAPIKeys = &cobra.Command{
|
|
Use: "list",
|
|
Short: "List the Api keys for headscale",
|
|
Aliases: []string{"ls", "show"},
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
output, _ := cmd.Flags().GetString("output")
|
|
|
|
ctx, client, conn, cancel := getHeadscaleCLIClient()
|
|
defer cancel()
|
|
defer conn.Close()
|
|
|
|
request := &v1.ListApiKeysRequest{}
|
|
|
|
response, err := client.ListApiKeys(ctx, request)
|
|
if err != nil {
|
|
ErrorOutput(
|
|
err,
|
|
fmt.Sprintf("Error getting the list of keys: %s", err),
|
|
output,
|
|
)
|
|
|
|
return
|
|
}
|
|
|
|
if output != "" {
|
|
SuccessOutput(response.ApiKeys, "", output)
|
|
|
|
return
|
|
}
|
|
|
|
tableData := pterm.TableData{
|
|
{"ID", "Prefix", "Expiration", "Created"},
|
|
}
|
|
for _, key := range response.ApiKeys {
|
|
expiration := "-"
|
|
|
|
if key.GetExpiration() != nil {
|
|
expiration = ColourTime(key.Expiration.AsTime())
|
|
}
|
|
|
|
tableData = append(tableData, []string{
|
|
strconv.FormatUint(key.GetId(), util.Base10),
|
|
key.GetPrefix(),
|
|
expiration,
|
|
key.GetCreatedAt().AsTime().Format(HeadscaleDateTimeFormat),
|
|
})
|
|
|
|
}
|
|
err = pterm.DefaultTable.WithHasHeader().WithData(tableData).Render()
|
|
if err != nil {
|
|
ErrorOutput(
|
|
err,
|
|
fmt.Sprintf("Failed to render pterm table: %s", err),
|
|
output,
|
|
)
|
|
|
|
return
|
|
}
|
|
},
|
|
}
|
|
|
|
var createAPIKeyCmd = &cobra.Command{
|
|
Use: "create",
|
|
Short: "Creates a new Api key",
|
|
Long: `
|
|
Creates a new Api key, the Api key is only visible on creation
|
|
and cannot be retrieved again.
|
|
If you loose a key, create a new one and revoke (expire) the old one.`,
|
|
Aliases: []string{"c", "new"},
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
output, _ := cmd.Flags().GetString("output")
|
|
|
|
log.Trace().
|
|
Msg("Preparing to create ApiKey")
|
|
|
|
request := &v1.CreateApiKeyRequest{}
|
|
|
|
durationStr, _ := cmd.Flags().GetString("expiration")
|
|
|
|
duration, err := model.ParseDuration(durationStr)
|
|
if err != nil {
|
|
ErrorOutput(
|
|
err,
|
|
fmt.Sprintf("Could not parse duration: %s\n", err),
|
|
output,
|
|
)
|
|
|
|
return
|
|
}
|
|
|
|
expiration := time.Now().UTC().Add(time.Duration(duration))
|
|
|
|
log.Trace().
|
|
Dur("expiration", time.Duration(duration)).
|
|
Msg("expiration has been set")
|
|
|
|
request.Expiration = timestamppb.New(expiration)
|
|
|
|
ctx, client, conn, cancel := getHeadscaleCLIClient()
|
|
defer cancel()
|
|
defer conn.Close()
|
|
|
|
response, err := client.CreateApiKey(ctx, request)
|
|
if err != nil {
|
|
ErrorOutput(
|
|
err,
|
|
fmt.Sprintf("Cannot create Api Key: %s\n", err),
|
|
output,
|
|
)
|
|
|
|
return
|
|
}
|
|
|
|
SuccessOutput(response.ApiKey, response.ApiKey, output)
|
|
},
|
|
}
|
|
|
|
var expireAPIKeyCmd = &cobra.Command{
|
|
Use: "expire",
|
|
Short: "Expire an ApiKey",
|
|
Aliases: []string{"revoke", "exp", "e"},
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
output, _ := cmd.Flags().GetString("output")
|
|
|
|
prefix, err := cmd.Flags().GetString("prefix")
|
|
if err != nil {
|
|
ErrorOutput(
|
|
err,
|
|
fmt.Sprintf("Error getting prefix from CLI flag: %s", err),
|
|
output,
|
|
)
|
|
|
|
return
|
|
}
|
|
|
|
ctx, client, conn, cancel := getHeadscaleCLIClient()
|
|
defer cancel()
|
|
defer conn.Close()
|
|
|
|
request := &v1.ExpireApiKeyRequest{
|
|
Prefix: prefix,
|
|
}
|
|
|
|
response, err := client.ExpireApiKey(ctx, request)
|
|
if err != nil {
|
|
ErrorOutput(
|
|
err,
|
|
fmt.Sprintf("Cannot expire Api Key: %s\n", err),
|
|
output,
|
|
)
|
|
|
|
return
|
|
}
|
|
|
|
SuccessOutput(response, "Key expired", output)
|
|
},
|
|
}
|