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 }