netmaker/netclient/local/routes.go

160 lines
4 KiB
Go
Raw Normal View History

2022-02-03 11:04:30 +08:00
package local
import (
"fmt"
2022-02-03 11:04:30 +08:00
"net"
"net/url"
2022-02-03 11:04:30 +08:00
"github.com/gravitl/netmaker/logger"
2022-02-03 11:04:30 +08:00
"github.com/gravitl/netmaker/netclient/ncutils"
"golang.zx2c4.com/wireguard/wgctrl/wgtypes"
)
// TODO handle ipv6 in future
2022-02-03 22:51:16 +08:00
// SetPeerRoutes - sets/removes ip routes for each peer on a network
2022-06-15 23:18:49 +08:00
func SetPeerRoutes(iface string, oldPeers map[string]bool, newPeers []wgtypes.PeerConfig) {
2022-08-23 04:44:04 +08:00
// get the default route
var hasRoute bool
gwIP, gwIface, err := GetDefaultRoute()
if err != nil {
logger.Log(0, "error getting default route:", err.Error())
}
if gwIP != "" && gwIface != "" && err == nil {
hasRoute = true
}
2022-02-03 11:04:30 +08:00
// traverse through all recieved peers
for _, peer := range newPeers {
2022-06-15 23:18:49 +08:00
for _, allowedIP := range peer.AllowedIPs {
if !oldPeers[allowedIP.String()] {
if err := setRoute(iface, &allowedIP, allowedIP.IP.String()); err != nil {
logger.Log(1, err.Error())
2022-02-03 11:04:30 +08:00
}
2022-06-15 23:18:49 +08:00
} else {
delete(oldPeers, allowedIP.String())
2022-02-03 11:04:30 +08:00
}
}
2022-08-23 05:46:09 +08:00
if peer.Endpoint == nil {
continue
}
2022-08-23 04:44:04 +08:00
if hasRoute && !ncutils.IpIsPrivate(peer.Endpoint.IP) {
ipNet, err := ncutils.GetIPNetFromString(peer.Endpoint.IP.String())
if err != nil {
logger.Log(0, "error parsing ip:", err.Error())
}
SetExplicitRoute(gwIface, &ipNet, gwIP)
2022-08-23 04:44:04 +08:00
}
2022-02-03 11:04:30 +08:00
}
2022-02-03 22:51:16 +08:00
// traverse through all remaining existing peers
2022-06-17 03:42:32 +08:00
for i := range oldPeers {
2022-06-15 23:18:49 +08:00
ip, err := ncutils.GetIPNetFromString(i)
if err != nil {
logger.Log(1, err.Error())
} else {
deleteRoute(iface, &ip, ip.IP.String())
2022-02-04 08:55:12 +08:00
}
}
}
// SetCurrentPeerRoutes - sets all the current peers
func SetCurrentPeerRoutes(iface, currentAddr string, peers []wgtypes.PeerConfig) {
2022-08-23 04:44:04 +08:00
// get the default route
var hasRoute bool
gwIP, gwIface, err := GetDefaultRoute()
if err != nil {
logger.Log(0, "error getting default route:", err.Error())
}
if gwIP != "" && gwIface != "" && err == nil {
hasRoute = true
}
// traverse through all recieved peers
2022-02-04 08:55:12 +08:00
for _, peer := range peers {
for _, allowedIP := range peer.AllowedIPs {
setRoute(iface, &allowedIP, currentAddr)
2022-02-03 11:04:30 +08:00
}
2022-08-23 05:46:09 +08:00
if peer.Endpoint == nil {
continue
}
2022-08-23 04:44:04 +08:00
if hasRoute && !ncutils.IpIsPrivate(peer.Endpoint.IP) {
ipNet, err := ncutils.GetIPNetFromString(peer.Endpoint.IP.String())
if err != nil {
logger.Log(0, "error parsing ip:", err.Error())
}
SetExplicitRoute(gwIface, &ipNet, gwIP)
2022-08-23 04:44:04 +08:00
}
2022-02-03 11:04:30 +08:00
}
2022-02-03 11:04:30 +08:00
}
2022-02-06 04:00:26 +08:00
// FlushPeerRoutes - removes all current peer routes
func FlushPeerRoutes(iface, currentAddr string, peers []wgtypes.Peer) {
2022-08-23 04:44:04 +08:00
// get the default route
var hasRoute bool
gwIP, gwIface, err := GetDefaultRoute()
if err != nil {
logger.Log(0, "error getting default route:", err.Error())
}
if gwIP != "" && gwIface != "" && err == nil {
hasRoute = true
}
2022-02-06 04:00:26 +08:00
for _, peer := range peers {
for _, allowedIP := range peer.AllowedIPs {
deleteRoute(iface, &allowedIP, currentAddr)
}
2022-08-23 04:44:04 +08:00
if hasRoute && !ncutils.IpIsPrivate(peer.Endpoint.IP) {
ipNet, err := ncutils.GetIPNetFromString(peer.Endpoint.IP.String())
if err != nil {
logger.Log(0, "error parsing ip:", err.Error())
}
deleteRoute(gwIface, &ipNet, gwIP)
}
2022-02-06 04:00:26 +08:00
}
}
// SetCIDRRoute - sets the CIDR route, used on join and restarts
func SetCIDRRoute(iface, currentAddr string, cidr *net.IPNet) {
setCidr(iface, currentAddr, cidr)
}
2022-02-06 04:00:26 +08:00
// RemoveCIDRRoute - removes a static cidr route
func RemoveCIDRRoute(iface, currentAddr string, cidr *net.IPNet) {
removeCidr(iface, cidr, currentAddr)
}
// SetNetmakerDomainRoute - sets explicit route over Gateway for a given DNS name
func SetNetmakerDomainRoute(domainRaw string) error {
var address net.IPNet
domain, err := url.Parse(domainRaw)
if err != nil {
return err
}
gwIP, gwIface, err := GetDefaultRoute()
if err != nil {
return fmt.Errorf("error getting default route: %w", err)
}
ips, err := net.LookupIP(domain.Hostname())
if err != nil {
return err
}
for _, ip := range ips {
if ipv4 := ip.To4(); ipv4 != nil {
address, err = ncutils.GetIPNetFromString(ipv4.String())
if err == nil {
break
}
}
}
if err != nil || address.IP == nil {
return fmt.Errorf("address not found")
}
return SetExplicitRoute(gwIface, &address, gwIP)
}