2021-04-28 10:55:29 -04:00
|
|
|
package headscale
|
|
|
|
|
|
|
|
import (
|
2022-05-30 09:31:06 -04:00
|
|
|
"fmt"
|
2022-09-01 18:06:19 -04:00
|
|
|
"net/netip"
|
2022-11-23 13:50:30 -05:00
|
|
|
|
|
|
|
"gorm.io/gorm"
|
2021-04-28 10:55:29 -04:00
|
|
|
)
|
|
|
|
|
2021-11-15 14:18:14 -05:00
|
|
|
const (
|
2022-07-29 11:35:21 -04:00
|
|
|
ErrRouteIsNotAvailable = Error("route is not available")
|
2021-11-15 14:18:14 -05:00
|
|
|
)
|
|
|
|
|
2022-11-23 13:50:30 -05:00
|
|
|
type Route struct {
|
|
|
|
gorm.Model
|
|
|
|
|
|
|
|
MachineID uint64
|
|
|
|
Machine Machine
|
|
|
|
Prefix IPPrefix
|
|
|
|
|
|
|
|
Advertised bool
|
|
|
|
Enabled bool
|
|
|
|
IsPrimary bool
|
|
|
|
}
|
|
|
|
|
2021-11-04 18:11:38 -04:00
|
|
|
// Deprecated: use machine function instead
|
2021-08-21 09:49:46 -04:00
|
|
|
// GetAdvertisedNodeRoutes returns the subnet routes advertised by a node (identified by
|
2021-11-13 03:39:04 -05:00
|
|
|
// namespace and node name).
|
2021-11-13 03:36:45 -05:00
|
|
|
func (h *Headscale) GetAdvertisedNodeRoutes(
|
|
|
|
namespace string,
|
|
|
|
nodeName string,
|
2022-09-01 18:06:19 -04:00
|
|
|
) (*[]netip.Prefix, error) {
|
2021-11-15 11:15:50 -05:00
|
|
|
machine, err := h.GetMachine(namespace, nodeName)
|
2021-04-28 10:55:29 -04:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2022-03-01 11:34:24 -05:00
|
|
|
return &machine.HostInfo.RoutableIPs, nil
|
2021-04-28 10:55:29 -04:00
|
|
|
}
|
|
|
|
|
2021-11-04 18:11:38 -04:00
|
|
|
// Deprecated: use machine function instead
|
2021-08-21 09:49:46 -04:00
|
|
|
// GetEnabledNodeRoutes returns the subnet routes enabled by a node (identified by
|
2021-11-13 03:39:04 -05:00
|
|
|
// namespace and node name).
|
2021-11-13 03:36:45 -05:00
|
|
|
func (h *Headscale) GetEnabledNodeRoutes(
|
|
|
|
namespace string,
|
|
|
|
nodeName string,
|
2022-09-01 18:06:19 -04:00
|
|
|
) ([]netip.Prefix, error) {
|
2021-11-15 11:15:50 -05:00
|
|
|
machine, err := h.GetMachine(namespace, nodeName)
|
2021-04-28 10:55:29 -04:00
|
|
|
if err != nil {
|
2021-05-08 07:59:18 -04:00
|
|
|
return nil, err
|
2021-04-28 10:55:29 -04:00
|
|
|
}
|
2021-08-21 09:49:46 -04:00
|
|
|
|
2022-03-01 11:34:24 -05:00
|
|
|
return machine.EnabledRoutes, nil
|
2021-08-21 09:49:46 -04:00
|
|
|
}
|
|
|
|
|
2021-11-04 18:11:38 -04:00
|
|
|
// Deprecated: use machine function instead
|
2021-11-13 03:39:04 -05:00
|
|
|
// IsNodeRouteEnabled checks if a certain route has been enabled.
|
2021-11-13 03:36:45 -05:00
|
|
|
func (h *Headscale) IsNodeRouteEnabled(
|
|
|
|
namespace string,
|
|
|
|
nodeName string,
|
|
|
|
routeStr string,
|
|
|
|
) bool {
|
2022-09-01 18:06:19 -04:00
|
|
|
route, err := netip.ParsePrefix(routeStr)
|
2021-08-21 09:49:46 -04:00
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
enabledRoutes, err := h.GetEnabledNodeRoutes(namespace, nodeName)
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, enabledRoute := range enabledRoutes {
|
|
|
|
if route == enabledRoute {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
2021-11-14 10:46:09 -05:00
|
|
|
|
2021-08-21 09:49:46 -04:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2021-11-04 18:11:38 -04:00
|
|
|
// Deprecated: use EnableRoute in machine.go
|
2021-08-21 09:49:46 -04:00
|
|
|
// EnableNodeRoute enables a subnet route advertised by a node (identified by
|
2021-11-13 03:39:04 -05:00
|
|
|
// namespace and node name).
|
2021-11-13 03:36:45 -05:00
|
|
|
func (h *Headscale) EnableNodeRoute(
|
|
|
|
namespace string,
|
|
|
|
nodeName string,
|
|
|
|
routeStr string,
|
|
|
|
) error {
|
2021-11-15 11:15:50 -05:00
|
|
|
machine, err := h.GetMachine(namespace, nodeName)
|
2021-08-21 09:49:46 -04:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-09-01 18:06:19 -04:00
|
|
|
route, err := netip.ParsePrefix(routeStr)
|
2021-08-21 09:49:46 -04:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
availableRoutes, err := h.GetAdvertisedNodeRoutes(namespace, nodeName)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-04-28 10:55:29 -04:00
|
|
|
|
2021-08-21 09:49:46 -04:00
|
|
|
enabledRoutes, err := h.GetEnabledNodeRoutes(namespace, nodeName)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
available := false
|
|
|
|
for _, availableRoute := range *availableRoutes {
|
|
|
|
// If the route is available, and not yet enabled, add it to the new routing table
|
|
|
|
if route == availableRoute {
|
|
|
|
available = true
|
|
|
|
if !h.IsNodeRouteEnabled(namespace, nodeName, routeStr) {
|
|
|
|
enabledRoutes = append(enabledRoutes, route)
|
2021-04-28 10:55:29 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-08-21 09:49:46 -04:00
|
|
|
|
|
|
|
if !available {
|
2022-07-29 11:35:21 -04:00
|
|
|
return ErrRouteIsNotAvailable
|
2021-08-21 09:49:46 -04:00
|
|
|
}
|
|
|
|
|
2022-03-01 11:34:24 -05:00
|
|
|
machine.EnabledRoutes = enabledRoutes
|
2022-05-30 09:31:06 -04:00
|
|
|
|
|
|
|
if err := h.db.Save(&machine).Error; err != nil {
|
|
|
|
return fmt.Errorf("failed to update node routes in the database: %w", err)
|
|
|
|
}
|
2021-08-21 09:49:46 -04:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|