2024-05-07 18:50:34 +02:00
|
|
|
//go:build (darwin && !ios) || dragonfly || freebsd || netbsd || openbsd
|
|
|
|
|
|
|
|
package networkmonitor
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2024-07-17 16:26:06 +02:00
|
|
|
"errors"
|
2024-05-07 18:50:34 +02:00
|
|
|
"fmt"
|
|
|
|
"syscall"
|
|
|
|
"unsafe"
|
|
|
|
|
|
|
|
log "github.com/sirupsen/logrus"
|
|
|
|
"golang.org/x/net/route"
|
|
|
|
"golang.org/x/sys/unix"
|
|
|
|
|
2024-06-13 13:24:24 +02:00
|
|
|
"github.com/netbirdio/netbird/client/internal/routemanager/systemops"
|
2024-05-07 18:50:34 +02:00
|
|
|
)
|
|
|
|
|
2024-06-13 13:24:24 +02:00
|
|
|
func checkChange(ctx context.Context, nexthopv4, nexthopv6 systemops.Nexthop, callback func()) error {
|
2024-05-07 18:50:34 +02:00
|
|
|
fd, err := unix.Socket(syscall.AF_ROUTE, syscall.SOCK_RAW, syscall.AF_UNSPEC)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("failed to open routing socket: %v", err)
|
|
|
|
}
|
|
|
|
defer func() {
|
2024-07-17 16:26:06 +02:00
|
|
|
err := unix.Close(fd)
|
|
|
|
if err != nil && !errors.Is(err, unix.EBADF) {
|
2024-05-07 18:50:34 +02:00
|
|
|
log.Errorf("Network monitor: failed to close routing socket: %v", err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2024-07-17 16:26:06 +02:00
|
|
|
go func() {
|
|
|
|
<-ctx.Done()
|
|
|
|
err := unix.Close(fd)
|
|
|
|
if err != nil && !errors.Is(err, unix.EBADF) {
|
|
|
|
log.Debugf("Network monitor: closed routing socket")
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2024-05-07 18:50:34 +02:00
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
2024-05-17 09:43:18 +02:00
|
|
|
return ErrStopped
|
2024-05-07 18:50:34 +02:00
|
|
|
default:
|
|
|
|
buf := make([]byte, 2048)
|
|
|
|
n, err := unix.Read(fd, buf)
|
|
|
|
if err != nil {
|
2024-07-17 16:26:06 +02:00
|
|
|
if !errors.Is(err, unix.EBADF) && !errors.Is(err, unix.EINVAL) {
|
|
|
|
log.Errorf("Network monitor: failed to read from routing socket: %v", err)
|
|
|
|
}
|
2024-05-07 18:50:34 +02:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
if n < unix.SizeofRtMsghdr {
|
|
|
|
log.Errorf("Network monitor: read from routing socket returned less than expected: %d bytes", n)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
msg := (*unix.RtMsghdr)(unsafe.Pointer(&buf[0]))
|
|
|
|
|
|
|
|
switch msg.Type {
|
|
|
|
// handle route changes
|
|
|
|
case unix.RTM_ADD, syscall.RTM_DELETE:
|
|
|
|
route, err := parseRouteMessage(buf[:n])
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("Network monitor: error parsing routing message: %v", err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if !route.Dst.Addr().IsUnspecified() {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
intf := "<nil>"
|
|
|
|
if route.Interface != nil {
|
|
|
|
intf = route.Interface.Name
|
|
|
|
}
|
|
|
|
switch msg.Type {
|
|
|
|
case unix.RTM_ADD:
|
|
|
|
log.Infof("Network monitor: default route changed: via %s, interface %s", route.Gw, intf)
|
2024-05-17 09:43:18 +02:00
|
|
|
go callback()
|
2024-05-07 18:50:34 +02:00
|
|
|
case unix.RTM_DELETE:
|
2024-06-13 13:24:24 +02:00
|
|
|
if nexthopv4.Intf != nil && route.Gw.Compare(nexthopv4.IP) == 0 || nexthopv6.Intf != nil && route.Gw.Compare(nexthopv6.IP) == 0 {
|
2024-05-07 18:50:34 +02:00
|
|
|
log.Infof("Network monitor: default route removed: via %s, interface %s", route.Gw, intf)
|
2024-05-17 09:43:18 +02:00
|
|
|
go callback()
|
2024-05-07 18:50:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-13 13:24:24 +02:00
|
|
|
func parseRouteMessage(buf []byte) (*systemops.Route, error) {
|
2024-05-07 18:50:34 +02:00
|
|
|
msgs, err := route.ParseRIB(route.RIBTypeRoute, buf)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("parse RIB: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(msgs) != 1 {
|
|
|
|
return nil, fmt.Errorf("unexpected RIB message msgs: %v", msgs)
|
|
|
|
}
|
|
|
|
|
|
|
|
msg, ok := msgs[0].(*route.RouteMessage)
|
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("unexpected RIB message type: %T", msgs[0])
|
|
|
|
}
|
|
|
|
|
2024-06-13 13:24:24 +02:00
|
|
|
return systemops.MsgToRoute(msg)
|
2024-05-07 18:50:34 +02:00
|
|
|
}
|