mirror of
https://github.com/juanfont/headscale.git
synced 2024-12-20 19:09:07 +01:00
770f3dcb93
* ensure valid tags is populated on user gets too Signed-off-by: Kristoffer Dalby <kristoffer@tailscale.com> * ensure forced tags are added Signed-off-by: Kristoffer Dalby <kristoffer@tailscale.com> * remove unused envvar in test Signed-off-by: Kristoffer Dalby <kristoffer@tailscale.com> * debug log auth/unauth tags in policy man Signed-off-by: Kristoffer Dalby <kristoffer@tailscale.com> * defer shutdown in tags test Signed-off-by: Kristoffer Dalby <kristoffer@tailscale.com> * add tag test with groups Signed-off-by: Kristoffer Dalby <kristoffer@tailscale.com> * add email, display name, picture to create user Updates #2166 Signed-off-by: Kristoffer Dalby <kristoffer@tailscale.com> * add ability to set display and email to cli Signed-off-by: Kristoffer Dalby <kristoffer@tailscale.com> * add email to test users in integration Signed-off-by: Kristoffer Dalby <kristoffer@tailscale.com> * fix issue where tags were only assigned to email, not username Fixes #2300 Fixes #2307 Signed-off-by: Kristoffer Dalby <kristoffer@tailscale.com> * expand principles to correct login name and if fix an issue where nodeip principles might not expand to all relevant IPs instead of taking the first in a prefix. Signed-off-by: Kristoffer Dalby <kristoffer@tailscale.com> * fix ssh unit test Signed-off-by: Kristoffer Dalby <kristoffer@tailscale.com> * update cli and oauth tests for users with email Signed-off-by: Kristoffer Dalby <kristoffer@tailscale.com> * index by test email Signed-off-by: Kristoffer Dalby <kristoffer@tailscale.com> * fix last test Signed-off-by: Kristoffer Dalby <kristoffer@tailscale.com> --------- Signed-off-by: Kristoffer Dalby <kristoffer@tailscale.com>
267 lines
6.0 KiB
Go
267 lines
6.0 KiB
Go
package cli
|
|
|
|
import (
|
|
"fmt"
|
|
"log"
|
|
"net/netip"
|
|
"strconv"
|
|
|
|
v1 "github.com/juanfont/headscale/gen/go/headscale/v1"
|
|
"github.com/pterm/pterm"
|
|
"github.com/spf13/cobra"
|
|
"google.golang.org/grpc/status"
|
|
"tailscale.com/net/tsaddr"
|
|
)
|
|
|
|
const (
|
|
Base10 = 10
|
|
)
|
|
|
|
func init() {
|
|
rootCmd.AddCommand(routesCmd)
|
|
listRoutesCmd.Flags().Uint64P("identifier", "i", 0, "Node identifier (ID)")
|
|
routesCmd.AddCommand(listRoutesCmd)
|
|
|
|
enableRouteCmd.Flags().Uint64P("route", "r", 0, "Route identifier (ID)")
|
|
err := enableRouteCmd.MarkFlagRequired("route")
|
|
if err != nil {
|
|
log.Fatal(err.Error())
|
|
}
|
|
routesCmd.AddCommand(enableRouteCmd)
|
|
|
|
disableRouteCmd.Flags().Uint64P("route", "r", 0, "Route identifier (ID)")
|
|
err = disableRouteCmd.MarkFlagRequired("route")
|
|
if err != nil {
|
|
log.Fatal(err.Error())
|
|
}
|
|
routesCmd.AddCommand(disableRouteCmd)
|
|
|
|
deleteRouteCmd.Flags().Uint64P("route", "r", 0, "Route identifier (ID)")
|
|
err = deleteRouteCmd.MarkFlagRequired("route")
|
|
if err != nil {
|
|
log.Fatal(err.Error())
|
|
}
|
|
routesCmd.AddCommand(deleteRouteCmd)
|
|
}
|
|
|
|
var routesCmd = &cobra.Command{
|
|
Use: "routes",
|
|
Short: "Manage the routes of Headscale",
|
|
Aliases: []string{"r", "route"},
|
|
}
|
|
|
|
var listRoutesCmd = &cobra.Command{
|
|
Use: "list",
|
|
Short: "List all routes",
|
|
Aliases: []string{"ls", "show"},
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
output, _ := cmd.Flags().GetString("output")
|
|
|
|
machineID, err := cmd.Flags().GetUint64("identifier")
|
|
if err != nil {
|
|
ErrorOutput(
|
|
err,
|
|
fmt.Sprintf("Error getting machine id from flag: %s", err),
|
|
output,
|
|
)
|
|
}
|
|
|
|
ctx, client, conn, cancel := newHeadscaleCLIWithConfig()
|
|
defer cancel()
|
|
defer conn.Close()
|
|
|
|
var routes []*v1.Route
|
|
|
|
if machineID == 0 {
|
|
response, err := client.GetRoutes(ctx, &v1.GetRoutesRequest{})
|
|
if err != nil {
|
|
ErrorOutput(
|
|
err,
|
|
fmt.Sprintf("Cannot get nodes: %s", status.Convert(err).Message()),
|
|
output,
|
|
)
|
|
}
|
|
|
|
if output != "" {
|
|
SuccessOutput(response.GetRoutes(), "", output)
|
|
}
|
|
|
|
routes = response.GetRoutes()
|
|
} else {
|
|
response, err := client.GetNodeRoutes(ctx, &v1.GetNodeRoutesRequest{
|
|
NodeId: machineID,
|
|
})
|
|
if err != nil {
|
|
ErrorOutput(
|
|
err,
|
|
fmt.Sprintf("Cannot get routes for node %d: %s", machineID, status.Convert(err).Message()),
|
|
output,
|
|
)
|
|
}
|
|
|
|
if output != "" {
|
|
SuccessOutput(response.GetRoutes(), "", output)
|
|
}
|
|
|
|
routes = response.GetRoutes()
|
|
}
|
|
|
|
tableData := routesToPtables(routes)
|
|
if err != nil {
|
|
ErrorOutput(err, fmt.Sprintf("Error converting to table: %s", err), output)
|
|
}
|
|
|
|
err = pterm.DefaultTable.WithHasHeader().WithData(tableData).Render()
|
|
if err != nil {
|
|
ErrorOutput(
|
|
err,
|
|
fmt.Sprintf("Failed to render pterm table: %s", err),
|
|
output,
|
|
)
|
|
}
|
|
},
|
|
}
|
|
|
|
var enableRouteCmd = &cobra.Command{
|
|
Use: "enable",
|
|
Short: "Set a route as enabled",
|
|
Long: `This command will make as enabled a given route.`,
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
output, _ := cmd.Flags().GetString("output")
|
|
|
|
routeID, err := cmd.Flags().GetUint64("route")
|
|
if err != nil {
|
|
ErrorOutput(
|
|
err,
|
|
fmt.Sprintf("Error getting machine id from flag: %s", err),
|
|
output,
|
|
)
|
|
}
|
|
|
|
ctx, client, conn, cancel := newHeadscaleCLIWithConfig()
|
|
defer cancel()
|
|
defer conn.Close()
|
|
|
|
response, err := client.EnableRoute(ctx, &v1.EnableRouteRequest{
|
|
RouteId: routeID,
|
|
})
|
|
if err != nil {
|
|
ErrorOutput(
|
|
err,
|
|
fmt.Sprintf("Cannot enable route %d: %s", routeID, status.Convert(err).Message()),
|
|
output,
|
|
)
|
|
}
|
|
|
|
if output != "" {
|
|
SuccessOutput(response, "", output)
|
|
}
|
|
},
|
|
}
|
|
|
|
var disableRouteCmd = &cobra.Command{
|
|
Use: "disable",
|
|
Short: "Set as disabled a given route",
|
|
Long: `This command will make as disabled a given route.`,
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
output, _ := cmd.Flags().GetString("output")
|
|
|
|
routeID, err := cmd.Flags().GetUint64("route")
|
|
if err != nil {
|
|
ErrorOutput(
|
|
err,
|
|
fmt.Sprintf("Error getting machine id from flag: %s", err),
|
|
output,
|
|
)
|
|
}
|
|
|
|
ctx, client, conn, cancel := newHeadscaleCLIWithConfig()
|
|
defer cancel()
|
|
defer conn.Close()
|
|
|
|
response, err := client.DisableRoute(ctx, &v1.DisableRouteRequest{
|
|
RouteId: routeID,
|
|
})
|
|
if err != nil {
|
|
ErrorOutput(
|
|
err,
|
|
fmt.Sprintf("Cannot disable route %d: %s", routeID, status.Convert(err).Message()),
|
|
output,
|
|
)
|
|
}
|
|
|
|
if output != "" {
|
|
SuccessOutput(response, "", output)
|
|
}
|
|
},
|
|
}
|
|
|
|
var deleteRouteCmd = &cobra.Command{
|
|
Use: "delete",
|
|
Short: "Delete a given route",
|
|
Long: `This command will delete a given route.`,
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
output, _ := cmd.Flags().GetString("output")
|
|
|
|
routeID, err := cmd.Flags().GetUint64("route")
|
|
if err != nil {
|
|
ErrorOutput(
|
|
err,
|
|
fmt.Sprintf("Error getting machine id from flag: %s", err),
|
|
output,
|
|
)
|
|
}
|
|
|
|
ctx, client, conn, cancel := newHeadscaleCLIWithConfig()
|
|
defer cancel()
|
|
defer conn.Close()
|
|
|
|
response, err := client.DeleteRoute(ctx, &v1.DeleteRouteRequest{
|
|
RouteId: routeID,
|
|
})
|
|
if err != nil {
|
|
ErrorOutput(
|
|
err,
|
|
fmt.Sprintf("Cannot delete route %d: %s", routeID, status.Convert(err).Message()),
|
|
output,
|
|
)
|
|
}
|
|
|
|
if output != "" {
|
|
SuccessOutput(response, "", output)
|
|
}
|
|
},
|
|
}
|
|
|
|
// routesToPtables converts the list of routes to a nice table.
|
|
func routesToPtables(routes []*v1.Route) pterm.TableData {
|
|
tableData := pterm.TableData{{"ID", "Node", "Prefix", "Advertised", "Enabled", "Primary"}}
|
|
|
|
for _, route := range routes {
|
|
var isPrimaryStr string
|
|
prefix, err := netip.ParsePrefix(route.GetPrefix())
|
|
if err != nil {
|
|
log.Printf("Error parsing prefix %s: %s", route.GetPrefix(), err)
|
|
|
|
continue
|
|
}
|
|
if tsaddr.IsExitRoute(prefix) {
|
|
isPrimaryStr = "-"
|
|
} else {
|
|
isPrimaryStr = strconv.FormatBool(route.GetIsPrimary())
|
|
}
|
|
|
|
tableData = append(tableData,
|
|
[]string{
|
|
strconv.FormatUint(route.GetId(), Base10),
|
|
route.GetNode().GetGivenName(),
|
|
route.GetPrefix(),
|
|
strconv.FormatBool(route.GetAdvertised()),
|
|
strconv.FormatBool(route.GetEnabled()),
|
|
isPrimaryStr,
|
|
})
|
|
}
|
|
|
|
return tableData
|
|
}
|