2022-11-05 08:07:22 +00:00
|
|
|
// nolint
|
2021-10-26 20:42:20 +00:00
|
|
|
package headscale
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2022-07-25 09:25:20 +00:00
|
|
|
"fmt"
|
2022-05-13 07:47:34 +00:00
|
|
|
"strings"
|
2021-11-04 22:19:27 +00:00
|
|
|
"time"
|
2021-10-26 20:42:20 +00:00
|
|
|
|
2022-04-15 11:11:41 +00:00
|
|
|
v1 "github.com/juanfont/headscale/gen/go/headscale/v1"
|
2021-11-04 22:19:27 +00:00
|
|
|
"github.com/rs/zerolog/log"
|
2022-05-13 08:17:52 +00:00
|
|
|
"google.golang.org/grpc/codes"
|
|
|
|
"google.golang.org/grpc/status"
|
2021-11-04 22:19:27 +00:00
|
|
|
"tailscale.com/tailcfg"
|
2022-11-05 08:07:22 +00:00
|
|
|
"tailscale.com/types/key"
|
2021-10-26 20:42:20 +00:00
|
|
|
)
|
|
|
|
|
2021-11-04 22:19:27 +00:00
|
|
|
type headscaleV1APIServer struct { // v1.HeadscaleServiceServer
|
|
|
|
v1.UnimplementedHeadscaleServiceServer
|
2021-10-26 20:42:20 +00:00
|
|
|
h *Headscale
|
|
|
|
}
|
|
|
|
|
2021-11-04 22:19:27 +00:00
|
|
|
func newHeadscaleV1APIServer(h *Headscale) v1.HeadscaleServiceServer {
|
2021-10-26 20:42:20 +00:00
|
|
|
return headscaleV1APIServer{
|
|
|
|
h: h,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-17 16:43:44 +00:00
|
|
|
func (api headscaleV1APIServer) GetUser(
|
2021-10-26 20:42:20 +00:00
|
|
|
ctx context.Context,
|
2023-01-17 16:43:44 +00:00
|
|
|
request *v1.GetUserRequest,
|
|
|
|
) (*v1.GetUserResponse, error) {
|
|
|
|
user, err := api.h.GetUser(request.GetName())
|
2021-11-04 22:19:27 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-10-29 16:44:32 +00:00
|
|
|
|
2023-01-17 16:43:44 +00:00
|
|
|
return &v1.GetUserResponse{User: user.toProto()}, nil
|
2021-10-29 16:44:32 +00:00
|
|
|
}
|
|
|
|
|
2023-01-17 16:43:44 +00:00
|
|
|
func (api headscaleV1APIServer) CreateUser(
|
2021-10-29 16:44:32 +00:00
|
|
|
ctx context.Context,
|
2023-01-17 16:43:44 +00:00
|
|
|
request *v1.CreateUserRequest,
|
|
|
|
) (*v1.CreateUserResponse, error) {
|
|
|
|
user, err := api.h.CreateUser(request.GetName())
|
2021-10-26 20:42:20 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2023-01-17 16:43:44 +00:00
|
|
|
return &v1.CreateUserResponse{User: user.toProto()}, nil
|
2021-11-04 22:19:27 +00:00
|
|
|
}
|
|
|
|
|
2023-01-17 16:43:44 +00:00
|
|
|
func (api headscaleV1APIServer) RenameUser(
|
2021-11-04 22:19:27 +00:00
|
|
|
ctx context.Context,
|
2023-01-17 16:43:44 +00:00
|
|
|
request *v1.RenameUserRequest,
|
|
|
|
) (*v1.RenameUserResponse, error) {
|
|
|
|
err := api.h.RenameUser(request.GetOldName(), request.GetNewName())
|
2021-11-04 22:19:27 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2023-01-17 16:43:44 +00:00
|
|
|
user, err := api.h.GetUser(request.GetNewName())
|
2021-11-04 22:19:27 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2023-01-17 16:43:44 +00:00
|
|
|
return &v1.RenameUserResponse{User: user.toProto()}, nil
|
2021-10-29 16:44:32 +00:00
|
|
|
}
|
|
|
|
|
2023-01-17 16:43:44 +00:00
|
|
|
func (api headscaleV1APIServer) DeleteUser(
|
2021-10-29 16:44:32 +00:00
|
|
|
ctx context.Context,
|
2023-01-17 16:43:44 +00:00
|
|
|
request *v1.DeleteUserRequest,
|
|
|
|
) (*v1.DeleteUserResponse, error) {
|
|
|
|
err := api.h.DestroyUser(request.GetName())
|
2021-10-29 16:44:32 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2023-01-17 16:43:44 +00:00
|
|
|
return &v1.DeleteUserResponse{}, nil
|
2021-10-29 16:44:32 +00:00
|
|
|
}
|
|
|
|
|
2023-01-17 16:43:44 +00:00
|
|
|
func (api headscaleV1APIServer) ListUsers(
|
2021-10-29 16:44:32 +00:00
|
|
|
ctx context.Context,
|
2023-01-17 16:43:44 +00:00
|
|
|
request *v1.ListUsersRequest,
|
|
|
|
) (*v1.ListUsersResponse, error) {
|
|
|
|
users, err := api.h.ListUsers()
|
2021-10-29 16:44:32 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2023-01-17 16:43:44 +00:00
|
|
|
response := make([]*v1.User, len(users))
|
|
|
|
for index, user := range users {
|
|
|
|
response[index] = user.toProto()
|
2021-11-04 22:19:27 +00:00
|
|
|
}
|
|
|
|
|
2023-01-17 16:43:44 +00:00
|
|
|
log.Trace().Caller().Interface("users", response).Msg("")
|
2021-11-04 22:19:27 +00:00
|
|
|
|
2023-01-17 16:43:44 +00:00
|
|
|
return &v1.ListUsersResponse{Users: response}, nil
|
2021-11-04 22:19:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (api headscaleV1APIServer) CreatePreAuthKey(
|
|
|
|
ctx context.Context,
|
|
|
|
request *v1.CreatePreAuthKeyRequest,
|
|
|
|
) (*v1.CreatePreAuthKeyResponse, error) {
|
2021-11-08 08:02:01 +00:00
|
|
|
var expiration time.Time
|
|
|
|
if request.GetExpiration() != nil {
|
|
|
|
expiration = request.GetExpiration().AsTime()
|
|
|
|
}
|
|
|
|
|
2022-09-23 07:58:06 +00:00
|
|
|
for _, tag := range request.AclTags {
|
|
|
|
err := validateTag(tag)
|
|
|
|
if err != nil {
|
|
|
|
return &v1.CreatePreAuthKeyResponse{
|
|
|
|
PreAuthKey: nil,
|
|
|
|
}, status.Error(codes.InvalidArgument, err.Error())
|
2022-08-25 10:43:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-04 22:19:27 +00:00
|
|
|
preAuthKey, err := api.h.CreatePreAuthKey(
|
2023-01-17 16:43:44 +00:00
|
|
|
request.GetUser(),
|
2021-11-08 20:49:03 +00:00
|
|
|
request.GetReusable(),
|
2021-11-04 22:19:27 +00:00
|
|
|
request.GetEphemeral(),
|
|
|
|
&expiration,
|
2022-08-25 10:03:38 +00:00
|
|
|
request.AclTags,
|
2021-11-04 22:19:27 +00:00
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &v1.CreatePreAuthKeyResponse{PreAuthKey: preAuthKey.toProto()}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (api headscaleV1APIServer) ExpirePreAuthKey(
|
|
|
|
ctx context.Context,
|
|
|
|
request *v1.ExpirePreAuthKeyRequest,
|
|
|
|
) (*v1.ExpirePreAuthKeyResponse, error) {
|
2023-01-17 16:43:44 +00:00
|
|
|
preAuthKey, err := api.h.GetPreAuthKey(request.GetUser(), request.Key)
|
2021-11-04 22:19:27 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = api.h.ExpirePreAuthKey(preAuthKey)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &v1.ExpirePreAuthKeyResponse{}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (api headscaleV1APIServer) ListPreAuthKeys(
|
|
|
|
ctx context.Context,
|
|
|
|
request *v1.ListPreAuthKeysRequest,
|
|
|
|
) (*v1.ListPreAuthKeysResponse, error) {
|
2023-01-17 16:43:44 +00:00
|
|
|
preAuthKeys, err := api.h.ListPreAuthKeys(request.GetUser())
|
2021-11-04 22:19:27 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
response := make([]*v1.PreAuthKey, len(preAuthKeys))
|
|
|
|
for index, key := range preAuthKeys {
|
|
|
|
response[index] = key.toProto()
|
|
|
|
}
|
|
|
|
|
|
|
|
return &v1.ListPreAuthKeysResponse{PreAuthKeys: response}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (api headscaleV1APIServer) RegisterMachine(
|
|
|
|
ctx context.Context,
|
|
|
|
request *v1.RegisterMachineRequest,
|
|
|
|
) (*v1.RegisterMachineResponse, error) {
|
2021-11-13 08:36:45 +00:00
|
|
|
log.Trace().
|
2023-01-17 16:43:44 +00:00
|
|
|
Str("user", request.GetUser()).
|
2022-08-10 13:35:26 +00:00
|
|
|
Str("node_key", request.GetKey()).
|
2021-11-13 08:36:45 +00:00
|
|
|
Msg("Registering machine")
|
2022-02-27 17:42:43 +00:00
|
|
|
|
2022-02-28 08:06:39 +00:00
|
|
|
machine, err := api.h.RegisterMachineFromAuthCallback(
|
2021-11-04 22:19:27 +00:00
|
|
|
request.GetKey(),
|
2023-01-17 16:43:44 +00:00
|
|
|
request.GetUser(),
|
2022-12-15 00:10:26 +00:00
|
|
|
nil,
|
2022-02-27 17:42:43 +00:00
|
|
|
RegisterMethodCLI,
|
2021-11-04 22:19:27 +00:00
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &v1.RegisterMachineResponse{Machine: machine.toProto()}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (api headscaleV1APIServer) GetMachine(
|
|
|
|
ctx context.Context,
|
|
|
|
request *v1.GetMachineRequest,
|
|
|
|
) (*v1.GetMachineResponse, error) {
|
|
|
|
machine, err := api.h.GetMachineByID(request.GetMachineId())
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &v1.GetMachineResponse{Machine: machine.toProto()}, nil
|
|
|
|
}
|
|
|
|
|
2022-04-25 19:16:14 +00:00
|
|
|
func (api headscaleV1APIServer) SetTags(
|
2022-04-15 11:11:41 +00:00
|
|
|
ctx context.Context,
|
2022-04-25 19:16:14 +00:00
|
|
|
request *v1.SetTagsRequest,
|
|
|
|
) (*v1.SetTagsResponse, error) {
|
|
|
|
machine, err := api.h.GetMachineByID(request.GetMachineId())
|
2022-04-15 11:11:41 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2022-05-13 07:47:34 +00:00
|
|
|
for _, tag := range request.GetTags() {
|
2022-07-25 09:25:20 +00:00
|
|
|
err := validateTag(tag)
|
|
|
|
if err != nil {
|
2022-05-13 08:17:52 +00:00
|
|
|
return &v1.SetTagsResponse{
|
2022-08-10 13:35:26 +00:00
|
|
|
Machine: nil,
|
2022-07-25 09:25:20 +00:00
|
|
|
}, status.Error(codes.InvalidArgument, err.Error())
|
2022-05-13 07:47:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-13 09:09:28 +00:00
|
|
|
err = api.h.SetTags(machine, request.GetTags())
|
|
|
|
if err != nil {
|
|
|
|
return &v1.SetTagsResponse{
|
|
|
|
Machine: nil,
|
|
|
|
}, status.Error(codes.Internal, err.Error())
|
|
|
|
}
|
2022-04-15 11:11:41 +00:00
|
|
|
|
2022-04-25 19:16:14 +00:00
|
|
|
log.Trace().
|
2022-05-16 19:41:46 +00:00
|
|
|
Str("machine", machine.Hostname).
|
2022-04-25 19:16:14 +00:00
|
|
|
Strs("tags", request.GetTags()).
|
|
|
|
Msg("Changing tags of machine")
|
2022-04-15 11:11:41 +00:00
|
|
|
|
2022-04-25 19:16:14 +00:00
|
|
|
return &v1.SetTagsResponse{Machine: machine.toProto()}, nil
|
2022-04-15 11:11:41 +00:00
|
|
|
}
|
|
|
|
|
2022-07-25 09:25:20 +00:00
|
|
|
func validateTag(tag string) error {
|
|
|
|
if strings.Index(tag, "tag:") != 0 {
|
|
|
|
return fmt.Errorf("tag must start with the string 'tag:'")
|
|
|
|
}
|
|
|
|
if strings.ToLower(tag) != tag {
|
|
|
|
return fmt.Errorf("tag should be lowercase")
|
|
|
|
}
|
|
|
|
if len(strings.Fields(tag)) > 1 {
|
|
|
|
return fmt.Errorf("tag should not contains space")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-11-04 22:19:27 +00:00
|
|
|
func (api headscaleV1APIServer) DeleteMachine(
|
|
|
|
ctx context.Context,
|
|
|
|
request *v1.DeleteMachineRequest,
|
|
|
|
) (*v1.DeleteMachineResponse, error) {
|
|
|
|
machine, err := api.h.GetMachineByID(request.GetMachineId())
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = api.h.DeleteMachine(
|
|
|
|
machine,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &v1.DeleteMachineResponse{}, nil
|
|
|
|
}
|
|
|
|
|
2021-11-21 13:40:19 +00:00
|
|
|
func (api headscaleV1APIServer) ExpireMachine(
|
|
|
|
ctx context.Context,
|
|
|
|
request *v1.ExpireMachineRequest,
|
|
|
|
) (*v1.ExpireMachineResponse, error) {
|
|
|
|
machine, err := api.h.GetMachineByID(request.GetMachineId())
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
api.h.ExpireMachine(
|
|
|
|
machine,
|
|
|
|
)
|
|
|
|
|
|
|
|
log.Trace().
|
2022-04-24 19:56:28 +00:00
|
|
|
Str("machine", machine.Hostname).
|
2021-11-21 13:40:19 +00:00
|
|
|
Time("expiry", *machine.Expiry).
|
|
|
|
Msg("machine expired")
|
|
|
|
|
|
|
|
return &v1.ExpireMachineResponse{Machine: machine.toProto()}, nil
|
|
|
|
}
|
|
|
|
|
2022-03-13 21:03:20 +00:00
|
|
|
func (api headscaleV1APIServer) RenameMachine(
|
|
|
|
ctx context.Context,
|
|
|
|
request *v1.RenameMachineRequest,
|
|
|
|
) (*v1.RenameMachineResponse, error) {
|
|
|
|
machine, err := api.h.GetMachineByID(request.GetMachineId())
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2022-04-24 19:56:28 +00:00
|
|
|
err = api.h.RenameMachine(
|
2022-03-13 21:03:20 +00:00
|
|
|
machine,
|
|
|
|
request.GetNewName(),
|
|
|
|
)
|
2022-04-24 19:56:28 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-03-13 21:03:20 +00:00
|
|
|
|
|
|
|
log.Trace().
|
2022-04-24 19:56:28 +00:00
|
|
|
Str("machine", machine.Hostname).
|
2022-05-16 18:35:35 +00:00
|
|
|
Str("new_name", request.GetNewName()).
|
|
|
|
Msg("machine renamed")
|
2022-03-13 21:03:20 +00:00
|
|
|
|
|
|
|
return &v1.RenameMachineResponse{Machine: machine.toProto()}, nil
|
|
|
|
}
|
|
|
|
|
2021-11-04 22:19:27 +00:00
|
|
|
func (api headscaleV1APIServer) ListMachines(
|
|
|
|
ctx context.Context,
|
|
|
|
request *v1.ListMachinesRequest,
|
|
|
|
) (*v1.ListMachinesResponse, error) {
|
2023-01-17 16:43:44 +00:00
|
|
|
if request.GetUser() != "" {
|
|
|
|
machines, err := api.h.ListMachinesByUser(request.GetUser())
|
2021-11-04 22:19:27 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
response := make([]*v1.Machine, len(machines))
|
|
|
|
for index, machine := range machines {
|
|
|
|
response[index] = machine.toProto()
|
|
|
|
}
|
|
|
|
|
|
|
|
return &v1.ListMachinesResponse{Machines: response}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
machines, err := api.h.ListMachines()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
response := make([]*v1.Machine, len(machines))
|
|
|
|
for index, machine := range machines {
|
2022-04-16 10:20:58 +00:00
|
|
|
m := machine.toProto()
|
2022-04-16 11:15:18 +00:00
|
|
|
validTags, invalidTags := getTags(
|
2022-05-04 20:56:55 +00:00
|
|
|
api.h.aclPolicy,
|
2022-04-16 11:15:18 +00:00
|
|
|
machine,
|
|
|
|
api.h.cfg.OIDC.StripEmaildomain,
|
|
|
|
)
|
2022-04-16 10:20:58 +00:00
|
|
|
m.InvalidTags = invalidTags
|
|
|
|
m.ValidTags = validTags
|
|
|
|
response[index] = m
|
2021-11-04 22:19:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return &v1.ListMachinesResponse{Machines: response}, nil
|
|
|
|
}
|
|
|
|
|
2022-05-01 13:44:34 +00:00
|
|
|
func (api headscaleV1APIServer) MoveMachine(
|
|
|
|
ctx context.Context,
|
|
|
|
request *v1.MoveMachineRequest,
|
|
|
|
) (*v1.MoveMachineResponse, error) {
|
|
|
|
machine, err := api.h.GetMachineByID(request.GetMachineId())
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2023-01-17 16:43:44 +00:00
|
|
|
err = api.h.SetMachineUser(machine, request.GetUser())
|
2022-05-01 13:44:34 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &v1.MoveMachineResponse{Machine: machine.toProto()}, nil
|
|
|
|
}
|
|
|
|
|
2022-11-26 00:03:39 +00:00
|
|
|
func (api headscaleV1APIServer) GetRoutes(
|
2021-11-04 22:19:27 +00:00
|
|
|
ctx context.Context,
|
2022-11-26 00:03:39 +00:00
|
|
|
request *v1.GetRoutesRequest,
|
|
|
|
) (*v1.GetRoutesResponse, error) {
|
|
|
|
routes, err := api.h.GetRoutes()
|
2021-11-04 22:19:27 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2022-11-26 00:03:39 +00:00
|
|
|
return &v1.GetRoutesResponse{
|
|
|
|
Routes: Routes(routes).toProto(),
|
2021-11-04 22:19:27 +00:00
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2022-11-26 00:03:39 +00:00
|
|
|
func (api headscaleV1APIServer) EnableRoute(
|
|
|
|
ctx context.Context,
|
|
|
|
request *v1.EnableRouteRequest,
|
|
|
|
) (*v1.EnableRouteResponse, error) {
|
|
|
|
err := api.h.EnableRoute(request.GetRouteId())
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &v1.EnableRouteResponse{}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (api headscaleV1APIServer) DisableRoute(
|
|
|
|
ctx context.Context,
|
|
|
|
request *v1.DisableRouteRequest,
|
|
|
|
) (*v1.DisableRouteResponse, error) {
|
|
|
|
err := api.h.DisableRoute(request.GetRouteId())
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &v1.DisableRouteResponse{}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (api headscaleV1APIServer) GetMachineRoutes(
|
2021-11-04 22:19:27 +00:00
|
|
|
ctx context.Context,
|
2022-11-26 00:03:39 +00:00
|
|
|
request *v1.GetMachineRoutesRequest,
|
|
|
|
) (*v1.GetMachineRoutesResponse, error) {
|
2021-11-04 22:19:27 +00:00
|
|
|
machine, err := api.h.GetMachineByID(request.GetMachineId())
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2022-11-26 00:03:39 +00:00
|
|
|
routes, err := api.h.GetMachineRoutes(machine)
|
2021-11-04 22:19:27 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2022-11-26 00:03:39 +00:00
|
|
|
return &v1.GetMachineRoutesResponse{
|
|
|
|
Routes: Routes(routes).toProto(),
|
2021-11-04 22:19:27 +00:00
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2023-03-06 08:05:40 +00:00
|
|
|
func (api headscaleV1APIServer) DeleteRoute(
|
|
|
|
ctx context.Context,
|
|
|
|
request *v1.DeleteRouteRequest,
|
|
|
|
) (*v1.DeleteRouteResponse, error) {
|
|
|
|
err := api.h.DeleteRoute(request.GetRouteId())
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &v1.DeleteRouteResponse{}, nil
|
|
|
|
}
|
|
|
|
|
2022-01-25 22:11:15 +00:00
|
|
|
func (api headscaleV1APIServer) CreateApiKey(
|
|
|
|
ctx context.Context,
|
|
|
|
request *v1.CreateApiKeyRequest,
|
|
|
|
) (*v1.CreateApiKeyResponse, error) {
|
|
|
|
var expiration time.Time
|
|
|
|
if request.GetExpiration() != nil {
|
|
|
|
expiration = request.GetExpiration().AsTime()
|
|
|
|
}
|
|
|
|
|
|
|
|
apiKey, _, err := api.h.CreateAPIKey(
|
|
|
|
&expiration,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &v1.CreateApiKeyResponse{ApiKey: apiKey}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (api headscaleV1APIServer) ExpireApiKey(
|
|
|
|
ctx context.Context,
|
|
|
|
request *v1.ExpireApiKeyRequest,
|
|
|
|
) (*v1.ExpireApiKeyResponse, error) {
|
|
|
|
var apiKey *APIKey
|
|
|
|
var err error
|
|
|
|
|
|
|
|
apiKey, err = api.h.GetAPIKey(request.Prefix)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = api.h.ExpireAPIKey(apiKey)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &v1.ExpireApiKeyResponse{}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (api headscaleV1APIServer) ListApiKeys(
|
|
|
|
ctx context.Context,
|
|
|
|
request *v1.ListApiKeysRequest,
|
|
|
|
) (*v1.ListApiKeysResponse, error) {
|
|
|
|
apiKeys, err := api.h.ListAPIKeys()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
response := make([]*v1.ApiKey, len(apiKeys))
|
|
|
|
for index, key := range apiKeys {
|
|
|
|
response[index] = key.toProto()
|
|
|
|
}
|
|
|
|
|
|
|
|
return &v1.ListApiKeysResponse{ApiKeys: response}, nil
|
|
|
|
}
|
|
|
|
|
2021-11-04 22:19:27 +00:00
|
|
|
// The following service calls are for testing and debugging
|
|
|
|
func (api headscaleV1APIServer) DebugCreateMachine(
|
|
|
|
ctx context.Context,
|
|
|
|
request *v1.DebugCreateMachineRequest,
|
|
|
|
) (*v1.DebugCreateMachineResponse, error) {
|
2023-01-17 16:43:44 +00:00
|
|
|
user, err := api.h.GetUser(request.GetUser())
|
2021-11-04 22:19:27 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2021-11-15 17:24:24 +00:00
|
|
|
routes, err := stringToIPPrefix(request.GetRoutes())
|
2021-11-04 22:19:27 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2021-11-13 08:36:45 +00:00
|
|
|
log.Trace().
|
|
|
|
Caller().
|
|
|
|
Interface("route-prefix", routes).
|
|
|
|
Interface("route-str", request.GetRoutes()).
|
|
|
|
Msg("")
|
2021-11-04 22:19:27 +00:00
|
|
|
|
|
|
|
hostinfo := tailcfg.Hostinfo{
|
|
|
|
RoutableIPs: routes,
|
|
|
|
OS: "TestOS",
|
|
|
|
Hostname: "DebugTestMachine",
|
|
|
|
}
|
|
|
|
|
2022-10-21 12:42:37 +00:00
|
|
|
givenName, err := api.h.GenerateGivenName(request.GetKey(), request.GetName())
|
2022-05-16 18:32:37 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2021-11-04 22:19:27 +00:00
|
|
|
newMachine := Machine{
|
|
|
|
MachineKey: request.GetKey(),
|
2022-04-24 19:56:28 +00:00
|
|
|
Hostname: request.GetName(),
|
2022-05-16 18:32:37 +00:00
|
|
|
GivenName: givenName,
|
2023-01-17 16:43:44 +00:00
|
|
|
User: *user,
|
2021-11-04 22:19:27 +00:00
|
|
|
|
|
|
|
Expiry: &time.Time{},
|
|
|
|
LastSeen: &time.Time{},
|
|
|
|
LastSuccessfulUpdate: &time.Time{},
|
|
|
|
|
2022-03-01 16:34:24 +00:00
|
|
|
HostInfo: HostInfo(hostinfo),
|
2021-11-04 22:19:27 +00:00
|
|
|
}
|
2022-11-28 15:54:23 +00:00
|
|
|
|
2022-11-05 08:07:22 +00:00
|
|
|
nodeKey := key.NodePublic{}
|
|
|
|
err = nodeKey.UnmarshalText([]byte(request.GetKey()))
|
|
|
|
if err != nil {
|
|
|
|
log.Panic().Msg("can not add machine for debug. invalid node key")
|
|
|
|
}
|
2021-11-04 22:19:27 +00:00
|
|
|
|
2022-02-28 16:34:50 +00:00
|
|
|
api.h.registrationCache.Set(
|
2022-11-05 08:07:22 +00:00
|
|
|
NodePublicKeyStripPrefix(nodeKey),
|
2022-02-28 16:34:50 +00:00
|
|
|
newMachine,
|
2022-02-28 22:42:30 +00:00
|
|
|
registerCacheExpiration,
|
2022-02-28 16:34:50 +00:00
|
|
|
)
|
2021-10-29 16:44:32 +00:00
|
|
|
|
2021-11-04 22:19:27 +00:00
|
|
|
return &v1.DebugCreateMachineResponse{Machine: newMachine.toProto()}, nil
|
2021-10-26 20:42:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (api headscaleV1APIServer) mustEmbedUnimplementedHeadscaleServiceServer() {}
|