2021-04-28 16:15:45 +02:00
|
|
|
package cli
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"log"
|
2022-09-03 23:46:14 +02:00
|
|
|
"net/netip"
|
2021-07-17 00:23:12 +02:00
|
|
|
"strconv"
|
2022-01-16 14:16:59 +01:00
|
|
|
"strings"
|
Fix nil dereference in nodes list command.
Fixes a nil pointer dereference observed when listing nodes that have
not yet connected.
```
panic: runtime error: invalid memory address or nil pointer dereference
[signal SIGSEGV: segmentation violation code=0x1 addr=0x0 pc=0xb931a4]
goroutine 1 [running]:
github.com/juanfont/headscale/cmd/headscale/cli.glob..func8(0x13c93e0, 0xc0004c4220, 0x0, 0x2)
/go/src/headscale/cmd/headscale/cli/nodes.go:74 +0x364
github.com/spf13/cobra.(*Command).execute(0x13c93e0, 0xc0004c41e0, 0x2, 0x2, 0x13c93e0, 0xc0004c41e0)
/go/pkg/mod/github.com/spf13/cobra@v1.1.3/command.go:856 +0x2c2
github.com/spf13/cobra.(*Command).ExecuteC(0x13ca2e0, 0xc000497110, 0xe76416, 0x6)
/go/pkg/mod/github.com/spf13/cobra@v1.1.3/command.go:960 +0x375
github.com/spf13/cobra.(*Command).Execute(...)
/go/pkg/mod/github.com/spf13/cobra@v1.1.3/command.go:897
main.main()
/go/src/headscale/cmd/headscale/headscale.go:89 +0x805
command terminated with exit code 2
```
2021-06-20 01:18:13 +02:00
|
|
|
"time"
|
2021-04-28 16:15:45 +02:00
|
|
|
|
2021-07-17 11:09:42 +02:00
|
|
|
survey "github.com/AlecAivazis/survey/v2"
|
2021-11-14 18:31:51 +01:00
|
|
|
"github.com/juanfont/headscale"
|
2021-11-04 23:44:35 +01:00
|
|
|
v1 "github.com/juanfont/headscale/gen/go/headscale/v1"
|
2021-08-15 23:10:39 +02:00
|
|
|
"github.com/pterm/pterm"
|
2021-04-28 16:15:45 +02:00
|
|
|
"github.com/spf13/cobra"
|
2021-11-07 11:15:32 +01:00
|
|
|
"google.golang.org/grpc/status"
|
2021-11-27 00:30:42 +01:00
|
|
|
"tailscale.com/types/key"
|
2021-04-28 16:15:45 +02:00
|
|
|
)
|
|
|
|
|
2021-07-31 23:14:24 +02:00
|
|
|
func init() {
|
|
|
|
rootCmd.AddCommand(nodeCmd)
|
2021-10-26 14:50:25 +02:00
|
|
|
listNodesCmd.Flags().StringP("namespace", "n", "", "Filter by namespace")
|
2022-05-13 11:46:28 +02:00
|
|
|
listNodesCmd.Flags().BoolP("tags", "t", false, "Show tags")
|
2021-10-24 23:02:57 +02:00
|
|
|
nodeCmd.AddCommand(listNodesCmd)
|
2021-10-27 23:51:42 +02:00
|
|
|
|
|
|
|
registerNodeCmd.Flags().StringP("namespace", "n", "", "Namespace")
|
2021-10-24 23:02:57 +02:00
|
|
|
err := registerNodeCmd.MarkFlagRequired("namespace")
|
2021-07-25 16:26:15 +02:00
|
|
|
if err != nil {
|
|
|
|
log.Fatalf(err.Error())
|
|
|
|
}
|
2021-10-28 15:30:41 +02:00
|
|
|
registerNodeCmd.Flags().StringP("key", "k", "", "Key")
|
|
|
|
err = registerNodeCmd.MarkFlagRequired("key")
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf(err.Error())
|
|
|
|
}
|
2021-07-25 15:04:06 +02:00
|
|
|
nodeCmd.AddCommand(registerNodeCmd)
|
2021-10-27 23:51:42 +02:00
|
|
|
|
2021-11-22 18:22:22 +01:00
|
|
|
expireNodeCmd.Flags().Uint64P("identifier", "i", 0, "Node identifier (ID)")
|
2021-11-21 14:40:44 +01:00
|
|
|
err = expireNodeCmd.MarkFlagRequired("identifier")
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf(err.Error())
|
|
|
|
}
|
|
|
|
nodeCmd.AddCommand(expireNodeCmd)
|
|
|
|
|
2022-03-13 22:03:20 +01:00
|
|
|
renameNodeCmd.Flags().Uint64P("identifier", "i", 0, "Node identifier (ID)")
|
|
|
|
err = renameNodeCmd.MarkFlagRequired("identifier")
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf(err.Error())
|
|
|
|
}
|
|
|
|
nodeCmd.AddCommand(renameNodeCmd)
|
|
|
|
|
2021-11-22 18:22:22 +01:00
|
|
|
deleteNodeCmd.Flags().Uint64P("identifier", "i", 0, "Node identifier (ID)")
|
2021-10-28 15:30:41 +02:00
|
|
|
err = deleteNodeCmd.MarkFlagRequired("identifier")
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf(err.Error())
|
|
|
|
}
|
2021-07-25 15:04:06 +02:00
|
|
|
nodeCmd.AddCommand(deleteNodeCmd)
|
2022-04-21 23:43:20 +02:00
|
|
|
|
2022-05-01 15:44:34 +02:00
|
|
|
moveNodeCmd.Flags().Uint64P("identifier", "i", 0, "Node identifier (ID)")
|
2022-05-02 06:32:33 +02:00
|
|
|
|
2022-05-01 15:44:34 +02:00
|
|
|
err = moveNodeCmd.MarkFlagRequired("identifier")
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf(err.Error())
|
|
|
|
}
|
2022-05-02 06:32:33 +02:00
|
|
|
|
2022-05-01 15:44:34 +02:00
|
|
|
moveNodeCmd.Flags().StringP("namespace", "n", "", "New namespace")
|
2022-05-02 06:32:33 +02:00
|
|
|
|
2022-05-01 15:44:34 +02:00
|
|
|
err = moveNodeCmd.MarkFlagRequired("namespace")
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf(err.Error())
|
|
|
|
}
|
|
|
|
nodeCmd.AddCommand(moveNodeCmd)
|
2022-05-03 20:35:28 +02:00
|
|
|
|
|
|
|
tagCmd.Flags().Uint64P("identifier", "i", 0, "Node identifier (ID)")
|
|
|
|
|
|
|
|
err = tagCmd.MarkFlagRequired("identifier")
|
2022-04-21 23:43:20 +02:00
|
|
|
if err != nil {
|
|
|
|
log.Fatalf(err.Error())
|
|
|
|
}
|
2022-05-04 22:56:55 +02:00
|
|
|
tagCmd.Flags().
|
|
|
|
StringSliceP("tags", "t", []string{}, "List of tags to add to the node")
|
2022-05-03 20:35:28 +02:00
|
|
|
nodeCmd.AddCommand(tagCmd)
|
2021-07-25 15:04:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
var nodeCmd = &cobra.Command{
|
2022-03-02 05:27:34 +01:00
|
|
|
Use: "nodes",
|
|
|
|
Short: "Manage the nodes of Headscale",
|
|
|
|
Aliases: []string{"node", "machine", "machines"},
|
2021-06-28 20:04:05 +02:00
|
|
|
}
|
|
|
|
|
2021-07-25 15:04:06 +02:00
|
|
|
var registerNodeCmd = &cobra.Command{
|
2021-10-28 15:30:41 +02:00
|
|
|
Use: "register",
|
2021-04-28 16:15:45 +02:00
|
|
|
Short: "Registers a machine to your network",
|
|
|
|
Run: func(cmd *cobra.Command, args []string) {
|
2021-11-04 23:44:35 +01:00
|
|
|
output, _ := cmd.Flags().GetString("output")
|
|
|
|
namespace, err := cmd.Flags().GetString("namespace")
|
2021-04-30 00:23:26 +02:00
|
|
|
if err != nil {
|
2021-11-04 23:44:35 +01:00
|
|
|
ErrorOutput(err, fmt.Sprintf("Error getting namespace: %s", err), output)
|
2021-11-14 16:46:09 +01:00
|
|
|
|
2021-11-04 23:44:35 +01:00
|
|
|
return
|
2021-04-30 00:23:26 +02:00
|
|
|
}
|
|
|
|
|
2021-11-07 10:41:14 +01:00
|
|
|
ctx, client, conn, cancel := getHeadscaleCLIClient()
|
2021-11-04 23:44:35 +01:00
|
|
|
defer cancel()
|
|
|
|
defer conn.Close()
|
|
|
|
|
|
|
|
machineKey, err := cmd.Flags().GetString("key")
|
2021-10-28 15:30:41 +02:00
|
|
|
if err != nil {
|
2021-11-13 09:36:45 +01:00
|
|
|
ErrorOutput(
|
|
|
|
err,
|
2022-08-10 15:35:26 +02:00
|
|
|
fmt.Sprintf("Error getting node key from flag: %s", err),
|
2021-11-13 09:36:45 +01:00
|
|
|
output,
|
|
|
|
)
|
2021-11-14 16:46:09 +01:00
|
|
|
|
2021-05-08 13:28:22 +02:00
|
|
|
return
|
|
|
|
}
|
2021-11-04 23:44:35 +01:00
|
|
|
|
|
|
|
request := &v1.RegisterMachineRequest{
|
|
|
|
Key: machineKey,
|
|
|
|
Namespace: namespace,
|
|
|
|
}
|
|
|
|
|
|
|
|
response, err := client.RegisterMachine(ctx, request)
|
2021-04-28 16:15:45 +02:00
|
|
|
if err != nil {
|
2021-11-13 09:36:45 +01:00
|
|
|
ErrorOutput(
|
|
|
|
err,
|
|
|
|
fmt.Sprintf(
|
|
|
|
"Cannot register machine: %s\n",
|
|
|
|
status.Convert(err).Message(),
|
|
|
|
),
|
|
|
|
output,
|
|
|
|
)
|
2021-11-14 16:46:09 +01:00
|
|
|
|
2021-04-28 16:15:45 +02:00
|
|
|
return
|
|
|
|
}
|
2021-11-04 23:44:35 +01:00
|
|
|
|
2022-11-13 22:40:57 +01:00
|
|
|
SuccessOutput(
|
|
|
|
response.Machine,
|
|
|
|
fmt.Sprintf("Machine %s registered", response.Machine.GivenName), output)
|
2021-04-28 16:15:45 +02:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2021-07-25 15:04:06 +02:00
|
|
|
var listNodesCmd = &cobra.Command{
|
2022-03-02 05:27:34 +01:00
|
|
|
Use: "list",
|
|
|
|
Short: "List nodes",
|
|
|
|
Aliases: []string{"ls", "show"},
|
2021-05-01 20:00:25 +02:00
|
|
|
Run: func(cmd *cobra.Command, args []string) {
|
2021-11-04 23:44:35 +01:00
|
|
|
output, _ := cmd.Flags().GetString("output")
|
|
|
|
namespace, err := cmd.Flags().GetString("namespace")
|
2021-05-01 20:00:25 +02:00
|
|
|
if err != nil {
|
2021-11-04 23:44:35 +01:00
|
|
|
ErrorOutput(err, fmt.Sprintf("Error getting namespace: %s", err), output)
|
2021-11-14 16:46:09 +01:00
|
|
|
|
2021-11-04 23:44:35 +01:00
|
|
|
return
|
2021-05-01 20:00:25 +02:00
|
|
|
}
|
2022-05-13 11:46:28 +02:00
|
|
|
showTags, err := cmd.Flags().GetBool("tags")
|
|
|
|
if err != nil {
|
|
|
|
ErrorOutput(err, fmt.Sprintf("Error getting tags flag: %s", err), output)
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
2021-05-01 20:00:25 +02:00
|
|
|
|
2021-11-07 10:41:14 +01:00
|
|
|
ctx, client, conn, cancel := getHeadscaleCLIClient()
|
2021-11-04 23:44:35 +01:00
|
|
|
defer cancel()
|
|
|
|
defer conn.Close()
|
2021-09-02 17:06:47 +02:00
|
|
|
|
2021-11-04 23:44:35 +01:00
|
|
|
request := &v1.ListMachinesRequest{
|
|
|
|
Namespace: namespace,
|
2021-09-03 10:23:45 +02:00
|
|
|
}
|
|
|
|
|
2021-11-04 23:44:35 +01:00
|
|
|
response, err := client.ListMachines(ctx, request)
|
|
|
|
if err != nil {
|
2021-11-13 09:36:45 +01:00
|
|
|
ErrorOutput(
|
|
|
|
err,
|
|
|
|
fmt.Sprintf("Cannot get nodes: %s", status.Convert(err).Message()),
|
|
|
|
output,
|
|
|
|
)
|
2021-11-14 16:46:09 +01:00
|
|
|
|
2021-05-08 13:58:51 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-11-04 23:44:35 +01:00
|
|
|
if output != "" {
|
|
|
|
SuccessOutput(response.Machines, "", output)
|
2021-11-14 16:46:09 +01:00
|
|
|
|
2021-11-04 23:44:35 +01:00
|
|
|
return
|
2021-05-01 20:00:25 +02:00
|
|
|
}
|
|
|
|
|
2022-05-13 11:46:28 +02:00
|
|
|
tableData, err := nodesToPtables(namespace, showTags, response.Machines)
|
2021-08-15 23:10:39 +02:00
|
|
|
if err != nil {
|
2021-11-04 23:44:35 +01:00
|
|
|
ErrorOutput(err, fmt.Sprintf("Error converting to table: %s", err), output)
|
2021-11-14 16:46:09 +01:00
|
|
|
|
2021-11-04 23:44:35 +01:00
|
|
|
return
|
2021-05-01 20:00:25 +02:00
|
|
|
}
|
|
|
|
|
2021-11-14 20:32:03 +01:00
|
|
|
err = pterm.DefaultTable.WithHasHeader().WithData(tableData).Render()
|
2021-08-15 23:35:03 +02:00
|
|
|
if err != nil {
|
2021-11-13 09:36:45 +01:00
|
|
|
ErrorOutput(
|
|
|
|
err,
|
|
|
|
fmt.Sprintf("Failed to render pterm table: %s", err),
|
|
|
|
output,
|
|
|
|
)
|
2021-11-14 16:46:09 +01:00
|
|
|
|
2021-11-04 23:44:35 +01:00
|
|
|
return
|
2021-08-15 23:35:03 +02:00
|
|
|
}
|
2021-05-01 20:00:25 +02:00
|
|
|
},
|
|
|
|
}
|
2021-07-17 00:23:12 +02:00
|
|
|
|
2021-11-21 14:40:44 +01:00
|
|
|
var expireNodeCmd = &cobra.Command{
|
|
|
|
Use: "expire",
|
|
|
|
Short: "Expire (log out) a machine in your network",
|
2021-11-21 22:35:36 +01:00
|
|
|
Long: "Expiring a node will keep the node in the database and force it to reauthenticate.",
|
2022-03-02 05:27:34 +01:00
|
|
|
Aliases: []string{"logout", "exp", "e"},
|
2021-11-21 14:40:44 +01:00
|
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
|
|
output, _ := cmd.Flags().GetString("output")
|
|
|
|
|
2021-11-21 22:34:03 +01:00
|
|
|
identifier, err := cmd.Flags().GetUint64("identifier")
|
2021-11-21 14:40:44 +01:00
|
|
|
if err != nil {
|
|
|
|
ErrorOutput(
|
|
|
|
err,
|
|
|
|
fmt.Sprintf("Error converting ID to integer: %s", err),
|
|
|
|
output,
|
|
|
|
)
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx, client, conn, cancel := getHeadscaleCLIClient()
|
|
|
|
defer cancel()
|
|
|
|
defer conn.Close()
|
|
|
|
|
|
|
|
request := &v1.ExpireMachineRequest{
|
2021-11-21 22:34:03 +01:00
|
|
|
MachineId: identifier,
|
2021-11-21 14:40:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
response, err := client.ExpireMachine(ctx, request)
|
|
|
|
if err != nil {
|
|
|
|
ErrorOutput(
|
|
|
|
err,
|
|
|
|
fmt.Sprintf(
|
|
|
|
"Cannot expire machine: %s\n",
|
|
|
|
status.Convert(err).Message(),
|
|
|
|
),
|
|
|
|
output,
|
|
|
|
)
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
SuccessOutput(response.Machine, "Machine expired", output)
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2022-03-13 22:03:20 +01:00
|
|
|
var renameNodeCmd = &cobra.Command{
|
2022-04-24 21:57:15 +02:00
|
|
|
Use: "rename NEW_NAME",
|
|
|
|
Short: "Renames a machine in your network",
|
2022-03-13 22:03:20 +01:00
|
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
|
|
output, _ := cmd.Flags().GetString("output")
|
|
|
|
|
|
|
|
identifier, err := cmd.Flags().GetUint64("identifier")
|
|
|
|
if err != nil {
|
|
|
|
ErrorOutput(
|
|
|
|
err,
|
|
|
|
fmt.Sprintf("Error converting ID to integer: %s", err),
|
|
|
|
output,
|
|
|
|
)
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx, client, conn, cancel := getHeadscaleCLIClient()
|
|
|
|
defer cancel()
|
|
|
|
defer conn.Close()
|
|
|
|
|
2022-03-13 22:10:41 +01:00
|
|
|
newName := ""
|
|
|
|
if len(args) > 0 {
|
|
|
|
newName = args[0]
|
|
|
|
}
|
2022-03-13 22:03:20 +01:00
|
|
|
request := &v1.RenameMachineRequest{
|
|
|
|
MachineId: identifier,
|
2022-04-24 21:57:15 +02:00
|
|
|
NewName: newName,
|
2022-03-13 22:03:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
response, err := client.RenameMachine(ctx, request)
|
|
|
|
if err != nil {
|
|
|
|
ErrorOutput(
|
|
|
|
err,
|
|
|
|
fmt.Sprintf(
|
2022-05-16 20:29:31 +02:00
|
|
|
"Cannot rename machine: %s\n",
|
2022-03-13 22:03:20 +01:00
|
|
|
status.Convert(err).Message(),
|
|
|
|
),
|
|
|
|
output,
|
|
|
|
)
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
SuccessOutput(response.Machine, "Machine renamed", output)
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2021-07-25 15:04:06 +02:00
|
|
|
var deleteNodeCmd = &cobra.Command{
|
2022-03-02 05:27:34 +01:00
|
|
|
Use: "delete",
|
|
|
|
Short: "Delete a node",
|
|
|
|
Aliases: []string{"del"},
|
2021-07-17 00:23:12 +02:00
|
|
|
Run: func(cmd *cobra.Command, args []string) {
|
2021-10-12 23:48:08 +02:00
|
|
|
output, _ := cmd.Flags().GetString("output")
|
2021-11-04 23:44:35 +01:00
|
|
|
|
2021-11-21 22:34:03 +01:00
|
|
|
identifier, err := cmd.Flags().GetUint64("identifier")
|
2021-07-17 00:23:12 +02:00
|
|
|
if err != nil {
|
2021-11-13 09:36:45 +01:00
|
|
|
ErrorOutput(
|
|
|
|
err,
|
|
|
|
fmt.Sprintf("Error converting ID to integer: %s", err),
|
|
|
|
output,
|
|
|
|
)
|
2021-11-14 16:46:09 +01:00
|
|
|
|
2021-11-04 23:44:35 +01:00
|
|
|
return
|
2021-07-17 00:23:12 +02:00
|
|
|
}
|
2021-11-04 23:44:35 +01:00
|
|
|
|
2021-11-07 10:41:14 +01:00
|
|
|
ctx, client, conn, cancel := getHeadscaleCLIClient()
|
2021-11-04 23:44:35 +01:00
|
|
|
defer cancel()
|
|
|
|
defer conn.Close()
|
|
|
|
|
|
|
|
getRequest := &v1.GetMachineRequest{
|
2021-11-21 22:34:03 +01:00
|
|
|
MachineId: identifier,
|
2021-11-04 23:44:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
getResponse, err := client.GetMachine(ctx, getRequest)
|
2021-07-17 00:23:12 +02:00
|
|
|
if err != nil {
|
2021-11-13 09:36:45 +01:00
|
|
|
ErrorOutput(
|
|
|
|
err,
|
|
|
|
fmt.Sprintf(
|
|
|
|
"Error getting node node: %s",
|
|
|
|
status.Convert(err).Message(),
|
|
|
|
),
|
|
|
|
output,
|
|
|
|
)
|
2021-11-14 16:46:09 +01:00
|
|
|
|
2021-11-04 23:44:35 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
deleteRequest := &v1.DeleteMachineRequest{
|
2021-11-22 18:22:22 +01:00
|
|
|
MachineId: identifier,
|
2021-07-17 00:23:12 +02:00
|
|
|
}
|
2021-07-17 11:09:42 +02:00
|
|
|
|
|
|
|
confirm := false
|
2021-10-16 12:30:52 +02:00
|
|
|
force, _ := cmd.Flags().GetBool("force")
|
|
|
|
if !force {
|
|
|
|
prompt := &survey.Confirm{
|
2021-11-13 09:36:45 +01:00
|
|
|
Message: fmt.Sprintf(
|
|
|
|
"Do you want to remove the node %s?",
|
|
|
|
getResponse.GetMachine().Name,
|
|
|
|
),
|
2021-10-16 12:30:52 +02:00
|
|
|
}
|
|
|
|
err = survey.AskOne(prompt, &confirm)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if confirm || force {
|
2021-11-04 23:44:35 +01:00
|
|
|
response, err := client.DeleteMachine(ctx, deleteRequest)
|
|
|
|
if output != "" {
|
|
|
|
SuccessOutput(response, "", output)
|
2021-11-14 16:46:09 +01:00
|
|
|
|
2021-10-12 23:48:08 +02:00
|
|
|
return
|
|
|
|
}
|
2021-07-17 11:09:42 +02:00
|
|
|
if err != nil {
|
2021-11-13 09:36:45 +01:00
|
|
|
ErrorOutput(
|
|
|
|
err,
|
|
|
|
fmt.Sprintf(
|
|
|
|
"Error deleting node: %s",
|
|
|
|
status.Convert(err).Message(),
|
|
|
|
),
|
|
|
|
output,
|
|
|
|
)
|
2021-11-14 16:46:09 +01:00
|
|
|
|
2021-10-12 23:48:08 +02:00
|
|
|
return
|
|
|
|
}
|
2021-11-13 09:36:45 +01:00
|
|
|
SuccessOutput(
|
|
|
|
map[string]string{"Result": "Node deleted"},
|
|
|
|
"Node deleted",
|
|
|
|
output,
|
|
|
|
)
|
2021-11-04 23:44:35 +01:00
|
|
|
} else {
|
|
|
|
SuccessOutput(map[string]string{"Result": "Node not deleted"}, "Node not deleted", output)
|
2021-07-17 00:23:12 +02:00
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
2021-08-15 23:10:39 +02:00
|
|
|
|
2022-05-01 15:44:34 +02:00
|
|
|
var moveNodeCmd = &cobra.Command{
|
|
|
|
Use: "move",
|
|
|
|
Short: "Move node to another namespace",
|
|
|
|
Aliases: []string{"mv"},
|
|
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
|
|
output, _ := cmd.Flags().GetString("output")
|
|
|
|
|
|
|
|
identifier, err := cmd.Flags().GetUint64("identifier")
|
|
|
|
if err != nil {
|
|
|
|
ErrorOutput(
|
|
|
|
err,
|
|
|
|
fmt.Sprintf("Error converting ID to integer: %s", err),
|
|
|
|
output,
|
|
|
|
)
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace, err := cmd.Flags().GetString("namespace")
|
|
|
|
if err != nil {
|
|
|
|
ErrorOutput(
|
|
|
|
err,
|
|
|
|
fmt.Sprintf("Error getting namespace: %s", err),
|
|
|
|
output,
|
|
|
|
)
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx, client, conn, cancel := getHeadscaleCLIClient()
|
|
|
|
defer cancel()
|
|
|
|
defer conn.Close()
|
|
|
|
|
|
|
|
getRequest := &v1.GetMachineRequest{
|
|
|
|
MachineId: identifier,
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err = client.GetMachine(ctx, getRequest)
|
|
|
|
if err != nil {
|
|
|
|
ErrorOutput(
|
|
|
|
err,
|
|
|
|
fmt.Sprintf(
|
|
|
|
"Error getting node: %s",
|
|
|
|
status.Convert(err).Message(),
|
|
|
|
),
|
|
|
|
output,
|
|
|
|
)
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
moveRequest := &v1.MoveMachineRequest{
|
|
|
|
MachineId: identifier,
|
|
|
|
Namespace: namespace,
|
|
|
|
}
|
|
|
|
|
|
|
|
moveResponse, err := client.MoveMachine(ctx, moveRequest)
|
|
|
|
if err != nil {
|
|
|
|
ErrorOutput(
|
|
|
|
err,
|
|
|
|
fmt.Sprintf(
|
|
|
|
"Error moving node: %s",
|
|
|
|
status.Convert(err).Message(),
|
|
|
|
),
|
|
|
|
output,
|
|
|
|
)
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-05-02 12:00:00 +02:00
|
|
|
SuccessOutput(moveResponse.Machine, "Node moved to another namespace", output)
|
2022-05-01 15:44:34 +02:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2021-11-13 09:36:45 +01:00
|
|
|
func nodesToPtables(
|
|
|
|
currentNamespace string,
|
2022-05-13 11:46:28 +02:00
|
|
|
showTags bool,
|
2021-11-13 09:36:45 +01:00
|
|
|
machines []*v1.Machine,
|
|
|
|
) (pterm.TableData, error) {
|
2022-05-13 11:46:28 +02:00
|
|
|
tableHeader := []string{
|
|
|
|
"ID",
|
2022-07-22 19:33:11 +02:00
|
|
|
"Hostname",
|
2022-05-13 11:46:28 +02:00
|
|
|
"Name",
|
2022-12-09 17:56:43 +01:00
|
|
|
"MachineKey",
|
2022-05-13 11:46:28 +02:00
|
|
|
"NodeKey",
|
|
|
|
"Namespace",
|
|
|
|
"IP addresses",
|
|
|
|
"Ephemeral",
|
|
|
|
"Last seen",
|
2023-01-11 13:35:47 +01:00
|
|
|
"Expiration",
|
2022-05-13 11:46:28 +02:00
|
|
|
"Online",
|
|
|
|
"Expired",
|
2021-11-13 09:36:45 +01:00
|
|
|
}
|
2022-05-13 11:46:28 +02:00
|
|
|
if showTags {
|
|
|
|
tableHeader = append(tableHeader, []string{
|
|
|
|
"ForcedTags",
|
|
|
|
"InvalidTags",
|
|
|
|
"ValidTags",
|
|
|
|
}...)
|
2021-11-13 09:36:45 +01:00
|
|
|
}
|
2022-05-13 11:46:28 +02:00
|
|
|
tableData := pterm.TableData{tableHeader}
|
2021-08-15 23:10:39 +02:00
|
|
|
|
2021-09-10 00:26:46 +02:00
|
|
|
for _, machine := range machines {
|
2021-08-15 23:10:39 +02:00
|
|
|
var ephemeral bool
|
2021-11-04 23:44:35 +01:00
|
|
|
if machine.PreAuthKey != nil && machine.PreAuthKey.Ephemeral {
|
2021-08-15 23:10:39 +02:00
|
|
|
ephemeral = true
|
|
|
|
}
|
2021-11-21 10:44:38 +01:00
|
|
|
|
2021-08-15 23:10:39 +02:00
|
|
|
var lastSeen time.Time
|
2021-09-01 23:44:42 +02:00
|
|
|
var lastSeenTime string
|
2021-09-10 00:37:01 +02:00
|
|
|
if machine.LastSeen != nil {
|
2021-11-04 23:44:35 +01:00
|
|
|
lastSeen = machine.LastSeen.AsTime()
|
2021-09-01 23:44:42 +02:00
|
|
|
lastSeenTime = lastSeen.Format("2006-01-02 15:04:05")
|
2021-08-15 23:10:39 +02:00
|
|
|
}
|
2021-11-21 10:44:38 +01:00
|
|
|
|
|
|
|
var expiry time.Time
|
2023-01-11 13:35:47 +01:00
|
|
|
var expiryTime string
|
2021-11-21 10:44:38 +01:00
|
|
|
if machine.Expiry != nil {
|
|
|
|
expiry = machine.Expiry.AsTime()
|
2023-01-11 13:35:47 +01:00
|
|
|
expiryTime = expiry.Format("2006-01-02 15:04:05")
|
|
|
|
} else {
|
|
|
|
expiryTime = "N/A"
|
2021-11-21 10:44:38 +01:00
|
|
|
}
|
|
|
|
|
2022-12-09 17:56:43 +01:00
|
|
|
var machineKey key.MachinePublic
|
|
|
|
err := machineKey.UnmarshalText(
|
|
|
|
[]byte(headscale.MachinePublicKeyEnsurePrefix(machine.MachineKey)),
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
machineKey = key.MachinePublic{}
|
|
|
|
}
|
|
|
|
|
2021-11-27 00:50:42 +01:00
|
|
|
var nodeKey key.NodePublic
|
2022-12-09 17:56:43 +01:00
|
|
|
err = nodeKey.UnmarshalText(
|
2021-11-27 21:25:12 +01:00
|
|
|
[]byte(headscale.NodePublicKeyEnsurePrefix(machine.NodeKey)),
|
|
|
|
)
|
2021-08-15 23:10:39 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var online string
|
2022-12-13 23:29:50 +01:00
|
|
|
if machine.Online {
|
2021-11-21 10:44:38 +01:00
|
|
|
online = pterm.LightGreen("online")
|
|
|
|
} else {
|
|
|
|
online = pterm.LightRed("offline")
|
|
|
|
}
|
|
|
|
|
|
|
|
var expired string
|
|
|
|
if expiry.IsZero() || expiry.After(time.Now()) {
|
|
|
|
expired = pterm.LightGreen("no")
|
2021-08-15 23:10:39 +02:00
|
|
|
} else {
|
2021-11-21 10:44:38 +01:00
|
|
|
expired = pterm.LightRed("yes")
|
2021-08-15 23:10:39 +02:00
|
|
|
}
|
2021-09-02 17:06:47 +02:00
|
|
|
|
2022-05-13 11:46:28 +02:00
|
|
|
var forcedTags string
|
2022-04-16 13:32:00 +02:00
|
|
|
for _, tag := range machine.ForcedTags {
|
2022-05-13 11:46:28 +02:00
|
|
|
forcedTags += "," + tag
|
2022-04-16 13:32:00 +02:00
|
|
|
}
|
2022-05-13 11:46:28 +02:00
|
|
|
forcedTags = strings.TrimLeft(forcedTags, ",")
|
|
|
|
var invalidTags string
|
2022-04-16 13:32:00 +02:00
|
|
|
for _, tag := range machine.InvalidTags {
|
2022-04-25 21:50:40 +02:00
|
|
|
if !contains(machine.ForcedTags, tag) {
|
2022-05-13 11:46:28 +02:00
|
|
|
invalidTags += "," + pterm.LightRed(tag)
|
2022-04-16 13:32:00 +02:00
|
|
|
}
|
|
|
|
}
|
2022-05-13 11:46:28 +02:00
|
|
|
invalidTags = strings.TrimLeft(invalidTags, ",")
|
|
|
|
var validTags string
|
2022-04-16 13:32:00 +02:00
|
|
|
for _, tag := range machine.ValidTags {
|
2022-04-25 21:50:40 +02:00
|
|
|
if !contains(machine.ForcedTags, tag) {
|
2022-05-13 11:46:28 +02:00
|
|
|
validTags += "," + pterm.LightGreen(tag)
|
2022-04-16 13:32:00 +02:00
|
|
|
}
|
|
|
|
}
|
2022-05-13 11:46:28 +02:00
|
|
|
validTags = strings.TrimLeft(validTags, ",")
|
2022-04-16 13:32:00 +02:00
|
|
|
|
2021-09-02 17:06:47 +02:00
|
|
|
var namespace string
|
2021-11-07 09:58:03 +01:00
|
|
|
if currentNamespace == "" || (currentNamespace == machine.Namespace.Name) {
|
2021-09-10 00:26:46 +02:00
|
|
|
namespace = pterm.LightMagenta(machine.Namespace.Name)
|
2021-09-02 17:06:47 +02:00
|
|
|
} else {
|
2021-10-24 23:02:57 +02:00
|
|
|
// Shared into this namespace
|
2021-09-10 00:26:46 +02:00
|
|
|
namespace = pterm.LightYellow(machine.Namespace.Name)
|
2021-09-02 17:06:47 +02:00
|
|
|
}
|
2022-05-08 21:06:12 +02:00
|
|
|
|
2022-05-16 14:59:46 +02:00
|
|
|
var IPV4Address string
|
|
|
|
var IPV6Address string
|
2022-05-08 21:21:10 +02:00
|
|
|
for _, addr := range machine.IpAddresses {
|
2022-09-03 23:46:14 +02:00
|
|
|
if netip.MustParseAddr(addr).Is4() {
|
2022-05-16 14:59:46 +02:00
|
|
|
IPV4Address = addr
|
2022-05-08 21:21:10 +02:00
|
|
|
} else {
|
2022-05-16 14:59:46 +02:00
|
|
|
IPV6Address = addr
|
2022-05-08 21:21:10 +02:00
|
|
|
}
|
2022-05-08 21:06:12 +02:00
|
|
|
}
|
|
|
|
|
2022-05-13 11:46:28 +02:00
|
|
|
nodeData := []string{
|
|
|
|
strconv.FormatUint(machine.Id, headscale.Base10),
|
|
|
|
machine.Name,
|
2022-07-22 19:33:11 +02:00
|
|
|
machine.GetGivenName(),
|
2022-12-09 17:56:43 +01:00
|
|
|
machineKey.ShortString(),
|
2022-05-13 11:46:28 +02:00
|
|
|
nodeKey.ShortString(),
|
|
|
|
namespace,
|
2022-05-16 14:59:46 +02:00
|
|
|
strings.Join([]string{IPV4Address, IPV6Address}, ", "),
|
2022-05-13 11:46:28 +02:00
|
|
|
strconv.FormatBool(ephemeral),
|
|
|
|
lastSeenTime,
|
2023-01-11 13:35:47 +01:00
|
|
|
expiryTime,
|
2022-05-13 11:46:28 +02:00
|
|
|
online,
|
|
|
|
expired,
|
|
|
|
}
|
|
|
|
if showTags {
|
|
|
|
nodeData = append(nodeData, []string{forcedTags, invalidTags, validTags}...)
|
|
|
|
}
|
2021-11-14 20:32:03 +01:00
|
|
|
tableData = append(
|
|
|
|
tableData,
|
2022-05-13 11:46:28 +02:00
|
|
|
nodeData,
|
2021-11-04 23:44:35 +01:00
|
|
|
)
|
2021-08-15 23:10:39 +02:00
|
|
|
}
|
2021-11-14 16:46:09 +01:00
|
|
|
|
2021-11-14 20:32:03 +01:00
|
|
|
return tableData, nil
|
2021-08-15 23:10:39 +02:00
|
|
|
}
|
2022-04-21 23:43:20 +02:00
|
|
|
|
|
|
|
var tagCmd = &cobra.Command{
|
2022-05-03 20:35:28 +02:00
|
|
|
Use: "tag",
|
|
|
|
Short: "Manage the tags of a node",
|
|
|
|
Aliases: []string{"tags", "t"},
|
2022-04-21 23:43:20 +02:00
|
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
|
|
output, _ := cmd.Flags().GetString("output")
|
|
|
|
ctx, client, conn, cancel := getHeadscaleCLIClient()
|
|
|
|
defer cancel()
|
|
|
|
defer conn.Close()
|
|
|
|
|
|
|
|
// retrieve flags from CLI
|
|
|
|
identifier, err := cmd.Flags().GetUint64("identifier")
|
|
|
|
if err != nil {
|
|
|
|
ErrorOutput(
|
|
|
|
err,
|
|
|
|
fmt.Sprintf("Error converting ID to integer: %s", err),
|
|
|
|
output,
|
|
|
|
)
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
2022-04-25 21:03:34 +02:00
|
|
|
tagsToSet, err := cmd.Flags().GetStringSlice("tags")
|
2022-04-21 23:43:20 +02:00
|
|
|
if err != nil {
|
|
|
|
ErrorOutput(
|
|
|
|
err,
|
|
|
|
fmt.Sprintf("Error retrieving list of tags to add to machine, %v", err),
|
|
|
|
output,
|
|
|
|
)
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-04-25 21:03:34 +02:00
|
|
|
// Sending tags to machine
|
|
|
|
request := &v1.SetTagsRequest{
|
2022-04-21 23:43:20 +02:00
|
|
|
MachineId: identifier,
|
2022-04-25 21:03:34 +02:00
|
|
|
Tags: tagsToSet,
|
2022-04-21 23:43:20 +02:00
|
|
|
}
|
2022-04-25 21:03:34 +02:00
|
|
|
resp, err := client.SetTags(ctx, request)
|
2022-04-21 23:43:20 +02:00
|
|
|
if err != nil {
|
|
|
|
ErrorOutput(
|
|
|
|
err,
|
2022-04-25 21:03:34 +02:00
|
|
|
fmt.Sprintf("Error while sending tags to headscale: %s", err),
|
2022-04-21 23:43:20 +02:00
|
|
|
output,
|
|
|
|
)
|
2022-05-13 10:17:52 +02:00
|
|
|
|
|
|
|
return
|
2022-04-21 23:43:20 +02:00
|
|
|
}
|
|
|
|
|
2022-04-25 21:03:34 +02:00
|
|
|
if resp != nil {
|
2022-04-21 23:43:20 +02:00
|
|
|
SuccessOutput(
|
2022-04-25 21:03:34 +02:00
|
|
|
resp.GetMachine(),
|
2022-04-21 23:43:20 +02:00
|
|
|
"Machine updated",
|
|
|
|
output,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|