Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: add receive interface information #35

Open
wants to merge 1 commit into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
44 changes: 34 additions & 10 deletions client.go
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,13 @@ type clientOpts struct {
ifaces []net.Interface
}

type Msg struct {
dns.Msg

ReceivedIfIndex int
ReceivedSrc net.Addr
}

// ClientOption fills the option struct to configure intefaces, etc.
type ClientOption func(*clientOpts)

Expand Down Expand Up @@ -168,7 +175,7 @@ var cleanupFreq = 10 * time.Second
// Start listeners and waits for the shutdown signal from exit channel
func (c *client) mainloop(ctx context.Context, params *lookupParams) {
// start listening for responses
msgCh := make(chan *dns.Msg, 32)
msgCh := make(chan *Msg, 32)
if c.ipv4conn != nil {
go c.recv(ctx, c.ipv4conn, msgCh)
}
Expand Down Expand Up @@ -218,6 +225,8 @@ func (c *client) mainloop(ctx context.Context, params *lookupParams) {
params.Service,
params.Domain)
}
entries[rr.Ptr].ReceivedIfIndex = msg.ReceivedIfIndex
entries[rr.Ptr].ReceivedSrc = msg.ReceivedSrc
entries[rr.Ptr].Expiry = now.Add(time.Duration(rr.Hdr.Ttl) * time.Second)
case *dns.SRV:
if params.ServiceInstanceName() != "" && params.ServiceInstanceName() != rr.Hdr.Name {
Expand All @@ -231,6 +240,8 @@ func (c *client) mainloop(ctx context.Context, params *lookupParams) {
params.Service,
params.Domain)
}
entries[rr.Hdr.Name].ReceivedIfIndex = msg.ReceivedIfIndex
entries[rr.Hdr.Name].ReceivedSrc = msg.ReceivedSrc
entries[rr.Hdr.Name].HostName = rr.Target
entries[rr.Hdr.Name].Port = int(rr.Port)
entries[rr.Hdr.Name].Expiry = now.Add(time.Duration(rr.Hdr.Ttl) * time.Second)
Expand All @@ -246,6 +257,8 @@ func (c *client) mainloop(ctx context.Context, params *lookupParams) {
params.Service,
params.Domain)
}
entries[rr.Hdr.Name].ReceivedIfIndex = msg.ReceivedIfIndex
entries[rr.Hdr.Name].ReceivedSrc = msg.ReceivedSrc
entries[rr.Hdr.Name].Text = rr.Txt
entries[rr.Hdr.Name].Expiry = now.Add(time.Duration(rr.Hdr.Ttl) * time.Second)
}
Expand Down Expand Up @@ -312,20 +325,28 @@ func (c *client) shutdown() {
}
}

// Data receiving routine reads from connection, unpacks packets into dns.Msg
// Data receiving routine reads from connection, unpacks packets into Msg
// structures and sends them to a given msgCh channel
func (c *client) recv(ctx context.Context, l interface{}, msgCh chan *dns.Msg) {
var readFrom func([]byte) (n int, src net.Addr, err error)
func (c *client) recv(ctx context.Context, l interface{}, msgCh chan *Msg) {
var readFrom func([]byte) (n int, ifIndex int, src net.Addr, err error)

switch pConn := l.(type) {
case *ipv6.PacketConn:
readFrom = func(b []byte) (n int, src net.Addr, err error) {
n, _, src, err = pConn.ReadFrom(b)
readFrom = func(b []byte) (n int, ifIndex int, src net.Addr, err error) {
var cm *ipv6.ControlMessage
n, cm, src, err = pConn.ReadFrom(b)
if cm != nil {
ifIndex = cm.IfIndex
}
return
}
case *ipv4.PacketConn:
readFrom = func(b []byte) (n int, src net.Addr, err error) {
n, _, src, err = pConn.ReadFrom(b)
readFrom = func(b []byte) (n int, ifIndex int, src net.Addr, err error) {
var cm *ipv4.ControlMessage
n, cm, src, err = pConn.ReadFrom(b)
if cm != nil {
ifIndex = cm.IfIndex
}
return
}

Expand All @@ -344,12 +365,15 @@ func (c *client) recv(ctx context.Context, l interface{}, msgCh chan *dns.Msg) {
return
}

n, _, err := readFrom(buf)
n, ifIndex, src, err := readFrom(buf)
if err != nil {
fatalErr = err
continue
}
msg := new(dns.Msg)
msg := &Msg{
ReceivedSrc: src,
ReceivedIfIndex: ifIndex,
}
if err := msg.Unpack(buf[:n]); err != nil {
// log.Printf("[WARN] mdns: Failed to unpack packet: %v", err)
continue
Expand Down
11 changes: 9 additions & 2 deletions connection.go
Original file line number Diff line number Diff line change
Expand Up @@ -110,9 +110,16 @@ func listMulticastInterfaces() []net.Interface {
if (ifi.Flags & net.FlagUp) == 0 {
continue
}
if (ifi.Flags & net.FlagMulticast) > 0 {
interfaces = append(interfaces, ifi)
if (ifi.Flags & net.FlagMulticast) == 0 {
continue
}

// Prevents bind of the bridge's slave interface.
addrs, err := ifi.Addrs()
if err != nil || len(addrs) == 0 {
continue
}
interfaces = append(interfaces, ifi)
}

return interfaces
Expand Down
3 changes: 3 additions & 0 deletions service.go
Original file line number Diff line number Diff line change
Expand Up @@ -110,6 +110,9 @@ type ServiceEntry struct {
Expiry time.Time `json:"expiry"` // Expiry of the service entry, will be converted to a TTL value
AddrIPv4 []net.IP `json:"-"` // Host machine IPv4 address
AddrIPv6 []net.IP `json:"-"` // Host machine IPv6 address

ReceivedIfIndex int
ReceivedSrc net.Addr
}

// newServiceEntry constructs a ServiceEntry.
Expand Down