forked from AdguardTeam/AdGuardHome
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge in DNS/adguard-home from 3443-fix-wired-dhcp to master Updates AdguardTeam#3443. Squashed commit of the following: commit ec7c3b7 Author: Eugene Burkov <E.Burkov@AdGuard.COM> Date: Tue Sep 7 18:21:07 2021 +0300 dhcpd: imp docs & naming commit d87c646 Author: Eugene Burkov <E.Burkov@AdGuard.COM> Date: Tue Sep 7 16:56:36 2021 +0300 dhcpd: fix build tags & log changes commit cbd0b3c Author: Eugene Burkov <E.Burkov@AdGuard.COM> Date: Tue Sep 7 16:26:50 2021 +0300 dhcpd: revert 3443 & imp broadcasting
- Loading branch information
1 parent
035d530
commit 570fe52
Showing
8 changed files
with
403 additions
and
16 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,37 @@ | ||
//go:build freebsd || openbsd | ||
// +build freebsd openbsd | ||
|
||
package dhcpd | ||
|
||
import ( | ||
"net" | ||
|
||
"github.com/AdguardTeam/golibs/log" | ||
"github.com/insomniacslk/dhcp/dhcpv4" | ||
) | ||
|
||
// broadcast sends resp to the broadcast address specific for network interface. | ||
func (s *v4Server) broadcast(peer net.Addr, conn net.PacketConn, resp *dhcpv4.DHCPv4) { | ||
// peer is expected to be of type *net.UDPConn as the server4.NewServer | ||
// initializes it. | ||
udpPeer, ok := peer.(*net.UDPAddr) | ||
if !ok { | ||
log.Error("dhcpv4: peer is of unexpected type %T", peer) | ||
|
||
return | ||
} | ||
|
||
// Despite the fact that server4.NewIPv4UDPConn explicitly sets socket | ||
// options to allow broadcasting, it also binds the connection to a | ||
// specific interface. On FreeBSD and OpenBSD conn.WriteTo causes | ||
// errors while writing to the addresses that belong to another | ||
// interface. So, use the broadcast address specific for the binded | ||
// interface. | ||
udpPeer.IP = s.conf.broadcastIP | ||
|
||
log.Debug("dhcpv4: sending to %s: %s", peer, resp.Summary()) | ||
|
||
if _, err := conn.WriteTo(resp.ToBytes(), peer); err != nil { | ||
log.Error("dhcpv4: conn.Write to %s failed: %s", peer, err) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,87 @@ | ||
//go:build freebsd || openbsd | ||
// +build freebsd openbsd | ||
|
||
package dhcpd | ||
|
||
import ( | ||
"bytes" | ||
"net" | ||
"testing" | ||
|
||
"github.com/AdguardTeam/golibs/netutil" | ||
"github.com/insomniacslk/dhcp/dhcpv4" | ||
"github.com/stretchr/testify/assert" | ||
"github.com/stretchr/testify/require" | ||
) | ||
|
||
func TestV4Server_Send_broadcast(t *testing.T) { | ||
b := &bytes.Buffer{} | ||
var peer *net.UDPAddr | ||
|
||
conn := &fakePacketConn{ | ||
writeTo: func(p []byte, addr net.Addr) (n int, err error) { | ||
udpPeer, ok := addr.(*net.UDPAddr) | ||
require.True(t, ok) | ||
|
||
peer = cloneUDPAddr(udpPeer) | ||
|
||
n, err = b.Write(p) | ||
require.NoError(t, err) | ||
|
||
return n, nil | ||
}, | ||
} | ||
|
||
defaultPeer := &net.UDPAddr{ | ||
IP: net.IP{1, 2, 3, 4}, | ||
// Use neither client nor server port. | ||
Port: 1234, | ||
} | ||
s := &v4Server{ | ||
conf: V4ServerConf{ | ||
broadcastIP: net.IP{1, 2, 3, 255}, | ||
}, | ||
} | ||
|
||
testCases := []struct { | ||
name string | ||
req *dhcpv4.DHCPv4 | ||
resp *dhcpv4.DHCPv4 | ||
}{{ | ||
name: "nak", | ||
req: &dhcpv4.DHCPv4{ | ||
GatewayIPAddr: netutil.IPv4Zero(), | ||
}, | ||
resp: &dhcpv4.DHCPv4{ | ||
Options: dhcpv4.OptionsFromList( | ||
dhcpv4.OptMessageType(dhcpv4.MessageTypeNak), | ||
), | ||
}, | ||
}, { | ||
name: "fully_unspecified", | ||
req: &dhcpv4.DHCPv4{ | ||
GatewayIPAddr: netutil.IPv4Zero(), | ||
ClientIPAddr: netutil.IPv4Zero(), | ||
}, | ||
resp: &dhcpv4.DHCPv4{ | ||
Options: dhcpv4.OptionsFromList( | ||
dhcpv4.OptMessageType(dhcpv4.MessageTypeOffer), | ||
), | ||
}, | ||
}} | ||
|
||
for _, tc := range testCases { | ||
t.Run(tc.name, func(t *testing.T) { | ||
s.send(cloneUDPAddr(defaultPeer), conn, tc.req, tc.resp) | ||
assert.EqualValues(t, tc.resp.ToBytes(), b.Bytes()) | ||
assert.Equal(t, &net.UDPAddr{ | ||
IP: s.conf.broadcastIP, | ||
Port: defaultPeer.Port, | ||
Zone: defaultPeer.Zone, | ||
}, peer) | ||
}) | ||
|
||
b.Reset() | ||
peer = nil | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,51 @@ | ||
//go:build aix || darwin || dragonfly || linux || netbsd || solaris | ||
// +build aix darwin dragonfly linux netbsd solaris | ||
|
||
package dhcpd | ||
|
||
import ( | ||
"net" | ||
|
||
"github.com/AdguardTeam/golibs/log" | ||
"github.com/insomniacslk/dhcp/dhcpv4" | ||
) | ||
|
||
// broadcast sends resp to the broadcast address specific for network interface. | ||
func (s *v4Server) broadcast(peer net.Addr, conn net.PacketConn, resp *dhcpv4.DHCPv4) { | ||
respData := resp.ToBytes() | ||
|
||
log.Debug("dhcpv4: sending to %s: %s", peer, resp.Summary()) | ||
|
||
// This write to 0xffffffff reverts some behavior changes made in | ||
// https://github.com/AdguardTeam/AdGuardHome/issues/3289. The DHCP | ||
// server should broadcast the message to 0xffffffff but it's | ||
// inconsistent with the actual mental model of DHCP implementation | ||
// which requires the network interface selection to bind to. | ||
// | ||
// See https://github.com/AdguardTeam/AdGuardHome/issues/3480 and | ||
// https://github.com/AdguardTeam/AdGuardHome/issues/3366. | ||
// | ||
// See also https://github.com/AdguardTeam/AdGuardHome/issues/3539. | ||
if _, err := conn.WriteTo(respData, peer); err != nil { | ||
log.Error("dhcpv4: conn.Write to %s failed: %s", peer, err) | ||
} | ||
|
||
// peer is expected to be of type *net.UDPConn as the server4.NewServer | ||
// initializes it. | ||
udpPeer, ok := peer.(*net.UDPAddr) | ||
if !ok { | ||
log.Error("dhcpv4: peer is of unexpected type %T", peer) | ||
|
||
return | ||
} | ||
|
||
// Broadcast the message one more time using the interface-specific | ||
// broadcast address. | ||
udpPeer.IP = s.conf.broadcastIP | ||
|
||
log.Debug("dhcpv4: sending to %s: %s", peer, resp.Summary()) | ||
|
||
if _, err := conn.WriteTo(respData, peer); err != nil { | ||
log.Error("dhcpv4: conn.Write to %s failed: %s", peer, err) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,96 @@ | ||
//go:build aix || darwin || dragonfly || linux || netbsd || solaris | ||
// +build aix darwin dragonfly linux netbsd solaris | ||
|
||
package dhcpd | ||
|
||
import ( | ||
"bytes" | ||
"net" | ||
"testing" | ||
|
||
"github.com/AdguardTeam/golibs/netutil" | ||
"github.com/insomniacslk/dhcp/dhcpv4" | ||
"github.com/stretchr/testify/assert" | ||
"github.com/stretchr/testify/require" | ||
) | ||
|
||
func TestV4Server_Send_broadcast(t *testing.T) { | ||
b := &bytes.Buffer{} | ||
var peers []*net.UDPAddr | ||
|
||
conn := &fakePacketConn{ | ||
writeTo: func(p []byte, addr net.Addr) (n int, err error) { | ||
udpPeer, ok := addr.(*net.UDPAddr) | ||
require.True(t, ok) | ||
|
||
peers = append(peers, cloneUDPAddr(udpPeer)) | ||
|
||
n, err = b.Write(p) | ||
require.NoError(t, err) | ||
|
||
return n, nil | ||
}, | ||
} | ||
|
||
defaultPeer := &net.UDPAddr{ | ||
IP: net.IP{1, 2, 3, 4}, | ||
// Use neither client nor server port. | ||
Port: 1234, | ||
} | ||
s := &v4Server{ | ||
conf: V4ServerConf{ | ||
broadcastIP: net.IP{1, 2, 3, 255}, | ||
}, | ||
} | ||
|
||
testCases := []struct { | ||
name string | ||
req *dhcpv4.DHCPv4 | ||
resp *dhcpv4.DHCPv4 | ||
}{{ | ||
name: "nak", | ||
req: &dhcpv4.DHCPv4{ | ||
GatewayIPAddr: netutil.IPv4Zero(), | ||
}, | ||
resp: &dhcpv4.DHCPv4{ | ||
Options: dhcpv4.OptionsFromList( | ||
dhcpv4.OptMessageType(dhcpv4.MessageTypeNak), | ||
), | ||
}, | ||
}, { | ||
name: "fully_unspecified", | ||
req: &dhcpv4.DHCPv4{ | ||
GatewayIPAddr: netutil.IPv4Zero(), | ||
ClientIPAddr: netutil.IPv4Zero(), | ||
}, | ||
resp: &dhcpv4.DHCPv4{ | ||
Options: dhcpv4.OptionsFromList( | ||
dhcpv4.OptMessageType(dhcpv4.MessageTypeOffer), | ||
), | ||
}, | ||
}} | ||
|
||
for _, tc := range testCases { | ||
t.Run(tc.name, func(t *testing.T) { | ||
s.send(cloneUDPAddr(defaultPeer), conn, tc.req, tc.resp) | ||
|
||
// The same response is written twice. | ||
respData := tc.resp.ToBytes() | ||
assert.EqualValues(t, append(respData, respData...), b.Bytes()) | ||
|
||
require.Len(t, peers, 2) | ||
|
||
assert.Equal(t, &net.UDPAddr{ | ||
IP: defaultPeer.IP, | ||
Port: defaultPeer.Port, | ||
}, peers[0]) | ||
assert.Equal(t, &net.UDPAddr{ | ||
IP: s.conf.broadcastIP, | ||
Port: defaultPeer.Port, | ||
}, peers[1]) | ||
}) | ||
|
||
b.Reset() | ||
peers = nil | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.