-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathnetlink.go
125 lines (106 loc) · 2.51 KB
/
netlink.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
package lldpd
import (
"net"
"syscall"
"github.com/jsimonetti/rtnetlink"
"github.com/mdlayher/netlink"
"github.com/pkg/errors"
"golang.org/x/sys/unix"
)
type nlListener struct {
Messages chan *linkMessage
list map[uint32]struct{}
log Logger
}
// NewNLListener listens on rtnetlink for addition and removal
// of interfaces and inform users on the Messages channel.
func NewNLListener(log Logger) *nlListener {
l := &nlListener{
Messages: make(chan *linkMessage, 64),
list: make(map[uint32]struct{}),
log: log,
}
return l
}
// Start will start the listener
func (l *nlListener) Start() {
go func() {
err := l.Listen()
if err != nil {
l.log.Error("msg", "could not listen", "error", err)
}
}()
}
// Listen will start the listener loop
func (l *nlListener) Listen() error {
nl, err := rtnetlink.Dial(&netlink.Config{Groups: unix.RTNLGRP_LINK})
if err != nil {
errors.Wrap(err, "could not dial rtnetlink")
}
//send request for current list of interfaces
req := &rtnetlink.LinkMessage{}
nl.Send(req, unix.RTM_GETLINK, netlink.Request|netlink.Dump)
for {
msgs, omsgs, err := nl.Receive()
if err != nil {
return errors.Wrap(err, "netlink receive error")
}
for i, msg := range msgs {
if m, ok := msg.(*rtnetlink.LinkMessage); ok {
if m.Type != syscall.ARPHRD_ETHER {
// skip non-ethernet
continue
}
if m.Family != syscall.AF_UNSPEC {
// skip non-generic
continue
}
if omsgs[i].Header.Type == unix.RTM_NEWLINK {
if _, ok := l.list[m.Index]; !ok {
link, _ := net.InterfaceByIndex(int(m.Index))
l.Messages <- &linkMessage{
ifi: link,
op: IF_ADD,
}
l.list[m.Index] = struct{}{}
l.log.Info("msg", "netlink reports new interface", "ifname", m.Attributes.Name, "ifindex", m.Index)
}
continue
}
if omsgs[i].Header.Type == unix.RTM_DELLINK {
if _, ok := l.list[m.Index]; ok {
l.Messages <- &linkMessage{
ifi: &net.Interface{
Index: int(m.Index),
Name: m.Attributes.Name,
},
op: IF_DEL,
}
delete(l.list, m.Index)
l.log.Info("msg", "netlink reports deleted interface", "ifname", m.Attributes.Name, "ifindex", m.Index)
}
continue
}
}
}
}
}
type linkOp uint8
const (
IF_ADD linkOp = 1
IF_DEL linkOp = 2
)
type linkMessage struct {
ifi *net.Interface
op linkOp
}
func (l linkOp) String() string {
switch l {
case IF_ADD:
return "ADD"
case IF_DEL:
return "DEL"
default:
return "UNKNOWN"
}
}