2024-05-07 18:50:34 +02:00
|
|
|
//go:build !android
|
|
|
|
|
|
|
|
package networkmonitor
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"syscall"
|
|
|
|
|
|
|
|
log "github.com/sirupsen/logrus"
|
|
|
|
"github.com/vishvananda/netlink"
|
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 {
|
|
|
|
if nexthopv4.Intf == nil && nexthopv6.Intf == nil {
|
2024-05-07 18:50:34 +02:00
|
|
|
return errors.New("no interfaces available")
|
|
|
|
}
|
|
|
|
|
|
|
|
linkChan := make(chan netlink.LinkUpdate)
|
|
|
|
done := make(chan struct{})
|
|
|
|
defer close(done)
|
|
|
|
|
|
|
|
if err := netlink.LinkSubscribe(linkChan, done); err != nil {
|
|
|
|
return fmt.Errorf("subscribe to link updates: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
routeChan := make(chan netlink.RouteUpdate)
|
|
|
|
if err := netlink.RouteSubscribe(routeChan, done); err != nil {
|
|
|
|
return fmt.Errorf("subscribe to route updates: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Info("Network monitor: started")
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
2024-05-17 09:43:18 +02:00
|
|
|
return ErrStopped
|
2024-05-07 18:50:34 +02:00
|
|
|
|
|
|
|
// handle interface state changes
|
|
|
|
case update := <-linkChan:
|
2024-06-13 13:24:24 +02:00
|
|
|
if (nexthopv4.Intf == nil || update.Index != int32(nexthopv4.Intf.Index)) && (nexthopv6.Intf == nil || update.Index != int32(nexthopv6.Intf.Index)) {
|
2024-05-07 18:50:34 +02:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
switch update.Header.Type {
|
|
|
|
case syscall.RTM_DELLINK:
|
|
|
|
log.Infof("Network monitor: monitored interface (%s) is gone", update.Link.Attrs().Name)
|
2024-05-17 09:43:18 +02:00
|
|
|
go callback()
|
2024-05-07 18:50:34 +02:00
|
|
|
return nil
|
|
|
|
case syscall.RTM_NEWLINK:
|
|
|
|
if (update.IfInfomsg.Flags&syscall.IFF_RUNNING) == 0 && update.Link.Attrs().OperState == netlink.OperDown {
|
|
|
|
log.Infof("Network monitor: monitored interface (%s) is down.", update.Link.Attrs().Name)
|
2024-05-17 09:43:18 +02:00
|
|
|
go callback()
|
2024-05-07 18:50:34 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// handle route changes
|
|
|
|
case route := <-routeChan:
|
|
|
|
// default route and main table
|
|
|
|
if route.Dst != nil || route.Table != syscall.RT_TABLE_MAIN {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
switch route.Type {
|
|
|
|
// triggered on added/replaced routes
|
|
|
|
case syscall.RTM_NEWROUTE:
|
|
|
|
log.Infof("Network monitor: default route changed: via %s, interface %d", route.Gw, route.LinkIndex)
|
2024-05-17 09:43:18 +02:00
|
|
|
go callback()
|
2024-05-07 18:50:34 +02:00
|
|
|
return nil
|
|
|
|
case syscall.RTM_DELROUTE:
|
2024-06-13 13:24:24 +02:00
|
|
|
if nexthopv4.Intf != nil && route.Gw.Equal(nexthopv4.IP.AsSlice()) || nexthopv6.Intf != nil && route.Gw.Equal(nexthopv6.IP.AsSlice()) {
|
2024-05-07 18:50:34 +02:00
|
|
|
log.Infof("Network monitor: default route removed: via %s, interface %d", route.Gw, route.LinkIndex)
|
2024-05-17 09:43:18 +02:00
|
|
|
go callback()
|
2024-05-07 18:50:34 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|