2baa33fb1f
Merge in DNS/adguard-home from 2240-removing-errorx-dependency to master Squashed commit of the following: commit 5bbe0567356f06e3b9ee5b3dc38d357b472cacb1 Merge: a6040850d02d16a0b4
Author: Eugene Burkov <e.burkov@adguard.com> Date: Thu Nov 5 14:32:22 2020 +0300 Merge branch 'master' into 2240-removing-errorx-dependency commit a6040850da3cefb131208097477b0956e80063fb Author: Eugene Burkov <e.burkov@adguard.com> Date: Thu Nov 5 14:23:36 2020 +0300 * dhcpd: convert some abbreviations to lowercase. commit d05bd51b994906b0ff52c5a8e779bd1f512f4bb7 Author: Eugene Burkov <e.burkov@adguard.com> Date: Thu Nov 5 12:47:20 2020 +0300 * agherr: last final fixes commit 164bca55035ff44e50b0abb33e129a0d24ffe87c Author: Ainar Garipov <A.Garipov@AdGuard.COM> Date: Tue Nov 3 19:11:10 2020 +0300 * all: final fixes again commit a0ac26f409c0b28a176cf2861d52c2f471b59484 Author: Ainar Garipov <A.Garipov@AdGuard.COM> Date: Tue Nov 3 18:51:39 2020 +0300 * all: final fixes commit 6147b02d402b513323b07e85856b348884f3a088 Merge: 9fd3af1a362cc334f4
Author: Eugene Burkov <e.burkov@adguard.com> Date: Tue Nov 3 18:26:03 2020 +0300 Merge branch 'master' into 2240-removing-errorx-dependency commit 9fd3af1a39a3189b5c41315a8ad1568ae5cb4fc9 Author: Eugene Burkov <e.burkov@adguard.com> Date: Tue Nov 3 18:23:08 2020 +0300 * all: remove useless helper commit 7cd9aeae639762b28b25f354d69c5cf74f670211 Author: Eugene Burkov <e.burkov@adguard.com> Date: Tue Nov 3 17:19:26 2020 +0300 * agherr: improved code tidiness commit a74a49236e9aaace070646dac710de9201105262 Merge: dc9dedbf2df34ee5c0
Author: Eugene Burkov <e.burkov@adguard.com> Date: Tue Nov 3 16:54:29 2020 +0300 Merge branch 'master' into 2240-removing-errorx-dependency commit dc9dedbf205756e3adaa3bc776d349bf3d8c69a5 Author: Eugene Burkov <e.burkov@adguard.com> Date: Tue Nov 3 16:40:08 2020 +0300 * agherr: improve and cover by tests commit fd6bfe9e282156cc60e006cb7cd46cce4d3a07a8 Author: Eugene Burkov <e.burkov@adguard.com> Date: Tue Nov 3 14:06:27 2020 +0300 * all: improve code quality commit ea00c2f8c5060e9611f9a80cfd0e4a039526d0c4 Author: Eugene Burkov <e.burkov@adguard.com> Date: Tue Nov 3 13:03:57 2020 +0300 * all: fix linter style warnings commit 8e75e1a681a7218c2b4c69adfa2b7e1e2966f9ac Author: Eugene Burkov <e.burkov@adguard.com> Date: Tue Nov 3 12:29:26 2020 +0300 * all: remove github.com/joomcode/errorx dependency Closes #2240.
216 lines
6.2 KiB
Go
216 lines
6.2 KiB
Go
// +build aix darwin dragonfly freebsd linux netbsd openbsd solaris
|
|
|
|
package dhcpd
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"net"
|
|
"os"
|
|
"runtime"
|
|
"time"
|
|
|
|
"github.com/AdguardTeam/AdGuardHome/internal/dhcpd/nclient4"
|
|
"github.com/AdguardTeam/golibs/log"
|
|
"github.com/insomniacslk/dhcp/dhcpv4"
|
|
"github.com/insomniacslk/dhcp/dhcpv6"
|
|
"github.com/insomniacslk/dhcp/dhcpv6/nclient6"
|
|
"github.com/insomniacslk/dhcp/iana"
|
|
)
|
|
|
|
// CheckIfOtherDHCPServersPresentV4 sends a DHCP request to the specified network interface,
|
|
// and waits for a response for a period defined by defaultDiscoverTime
|
|
func CheckIfOtherDHCPServersPresentV4(ifaceName string) (bool, error) {
|
|
iface, err := net.InterfaceByName(ifaceName)
|
|
if err != nil {
|
|
return false, fmt.Errorf("couldn't find interface by name %s: %w", ifaceName, err)
|
|
}
|
|
|
|
// get ipv4 address of an interface
|
|
ifaceIPNet := getIfaceIPv4(*iface)
|
|
if len(ifaceIPNet) == 0 {
|
|
return false, fmt.Errorf("couldn't find IPv4 address of interface %s %+v", ifaceName, iface)
|
|
}
|
|
|
|
if runtime.GOOS == "darwin" {
|
|
return false, fmt.Errorf("can't find DHCP server: not supported on macOS")
|
|
}
|
|
|
|
srcIP := ifaceIPNet[0]
|
|
src := net.JoinHostPort(srcIP.String(), "68")
|
|
dst := "255.255.255.255:67"
|
|
|
|
hostname, _ := os.Hostname()
|
|
|
|
req, err := dhcpv4.NewDiscovery(iface.HardwareAddr)
|
|
if err != nil {
|
|
return false, fmt.Errorf("dhcpv4.NewDiscovery: %w", err)
|
|
}
|
|
req.Options.Update(dhcpv4.OptClientIdentifier(iface.HardwareAddr))
|
|
req.Options.Update(dhcpv4.OptHostName(hostname))
|
|
|
|
// resolve 0.0.0.0:68
|
|
udpAddr, err := net.ResolveUDPAddr("udp4", src)
|
|
if err != nil {
|
|
return false, fmt.Errorf("couldn't resolve UDP address %s: %w", src, err)
|
|
}
|
|
|
|
if !udpAddr.IP.To4().Equal(srcIP) {
|
|
return false, fmt.Errorf("resolved UDP address is not %s: %w", src, err)
|
|
}
|
|
|
|
// resolve 255.255.255.255:67
|
|
dstAddr, err := net.ResolveUDPAddr("udp4", dst)
|
|
if err != nil {
|
|
return false, fmt.Errorf("couldn't resolve UDP address %s: %w", dst, err)
|
|
}
|
|
|
|
// bind to 0.0.0.0:68
|
|
log.Tracef("Listening to udp4 %+v", udpAddr)
|
|
c, err := nclient4.NewRawUDPConn(ifaceName, 68)
|
|
if err != nil {
|
|
return false, fmt.Errorf("couldn't listen on :68: %w", err)
|
|
}
|
|
if c != nil {
|
|
defer c.Close()
|
|
}
|
|
|
|
// send to 255.255.255.255:67
|
|
_, err = c.WriteTo(req.ToBytes(), dstAddr)
|
|
if err != nil {
|
|
return false, fmt.Errorf("couldn't send a packet to %s: %w", dst, err)
|
|
}
|
|
|
|
for {
|
|
// wait for answer
|
|
log.Tracef("Waiting %v for an answer", defaultDiscoverTime)
|
|
// TODO: replicate dhclient's behaviour of retrying several times with progressively bigger timeouts
|
|
b := make([]byte, 1500)
|
|
_ = c.SetReadDeadline(time.Now().Add(defaultDiscoverTime))
|
|
n, _, err := c.ReadFrom(b)
|
|
if isTimeout(err) {
|
|
// timed out -- no DHCP servers
|
|
log.Debug("DHCPv4: didn't receive DHCP response")
|
|
return false, nil
|
|
}
|
|
if err != nil {
|
|
return false, fmt.Errorf("couldn't receive packet: %w", err)
|
|
}
|
|
|
|
log.Tracef("Received packet (%v bytes)", n)
|
|
|
|
response, err := dhcpv4.FromBytes(b[:n])
|
|
if err != nil {
|
|
log.Debug("DHCPv4: dhcpv4.FromBytes: %s", err)
|
|
continue
|
|
}
|
|
|
|
log.Debug("DHCPv4: received message from server: %s", response.Summary())
|
|
|
|
if !(response.OpCode == dhcpv4.OpcodeBootReply &&
|
|
response.HWType == iana.HWTypeEthernet &&
|
|
bytes.Equal(response.ClientHWAddr, iface.HardwareAddr) &&
|
|
bytes.Equal(response.TransactionID[:], req.TransactionID[:]) &&
|
|
response.Options.Has(dhcpv4.OptionDHCPMessageType)) {
|
|
log.Debug("DHCPv4: received message from server doesn't match our request")
|
|
continue
|
|
}
|
|
|
|
log.Tracef("The packet is from an active DHCP server")
|
|
// that's a DHCP server there
|
|
return true, nil
|
|
}
|
|
}
|
|
|
|
// CheckIfOtherDHCPServersPresentV6 sends a DHCP request to the specified network interface,
|
|
// and waits for a response for a period defined by defaultDiscoverTime
|
|
func CheckIfOtherDHCPServersPresentV6(ifaceName string) (bool, error) {
|
|
iface, err := net.InterfaceByName(ifaceName)
|
|
if err != nil {
|
|
return false, fmt.Errorf("dhcpv6: net.InterfaceByName: %s: %w", ifaceName, err)
|
|
}
|
|
|
|
ifaceIPNet := getIfaceIPv6(*iface)
|
|
if len(ifaceIPNet) == 0 {
|
|
return false, fmt.Errorf("dhcpv6: couldn't find IPv6 address of interface %s %+v", ifaceName, iface)
|
|
}
|
|
|
|
srcIP := ifaceIPNet[0]
|
|
src := net.JoinHostPort(srcIP.String(), "546")
|
|
dst := "[ff02::1:2]:547"
|
|
|
|
req, err := dhcpv6.NewSolicit(iface.HardwareAddr)
|
|
if err != nil {
|
|
return false, fmt.Errorf("dhcpv6: dhcpv6.NewSolicit: %w", err)
|
|
}
|
|
|
|
udpAddr, err := net.ResolveUDPAddr("udp6", src)
|
|
if err != nil {
|
|
return false, fmt.Errorf("dhcpv6: Couldn't resolve UDP address %s: %w", src, err)
|
|
}
|
|
|
|
if !udpAddr.IP.To16().Equal(srcIP) {
|
|
return false, fmt.Errorf("dhcpv6: Resolved UDP address is not %s: %w", src, err)
|
|
}
|
|
|
|
dstAddr, err := net.ResolveUDPAddr("udp6", dst)
|
|
if err != nil {
|
|
return false, fmt.Errorf("dhcpv6: Couldn't resolve UDP address %s: %w", dst, err)
|
|
}
|
|
|
|
log.Debug("DHCPv6: Listening to udp6 %+v", udpAddr)
|
|
c, err := nclient6.NewIPv6UDPConn(ifaceName, dhcpv6.DefaultClientPort)
|
|
if err != nil {
|
|
return false, fmt.Errorf("dhcpv6: Couldn't listen on :546: %w", err)
|
|
}
|
|
if c != nil {
|
|
defer c.Close()
|
|
}
|
|
|
|
_, err = c.WriteTo(req.ToBytes(), dstAddr)
|
|
if err != nil {
|
|
return false, fmt.Errorf("dhcpv6: Couldn't send a packet to %s: %w", dst, err)
|
|
}
|
|
|
|
for {
|
|
log.Debug("DHCPv6: Waiting %v for an answer", defaultDiscoverTime)
|
|
b := make([]byte, 4096)
|
|
_ = c.SetReadDeadline(time.Now().Add(defaultDiscoverTime))
|
|
n, _, err := c.ReadFrom(b)
|
|
if isTimeout(err) {
|
|
log.Debug("DHCPv6: didn't receive DHCP response")
|
|
return false, nil
|
|
}
|
|
if err != nil {
|
|
return false, fmt.Errorf("couldn't receive packet: %w", err)
|
|
}
|
|
|
|
log.Debug("DHCPv6: Received packet (%v bytes)", n)
|
|
|
|
resp, err := dhcpv6.FromBytes(b[:n])
|
|
if err != nil {
|
|
log.Debug("DHCPv6: dhcpv6.FromBytes: %s", err)
|
|
continue
|
|
}
|
|
|
|
log.Debug("DHCPv6: received message from server: %s", resp.Summary())
|
|
|
|
cid := req.Options.ClientID()
|
|
msg, err := resp.GetInnerMessage()
|
|
if err != nil {
|
|
log.Debug("DHCPv6: resp.GetInnerMessage: %s", err)
|
|
continue
|
|
}
|
|
rcid := msg.Options.ClientID()
|
|
if resp.Type() == dhcpv6.MessageTypeAdvertise &&
|
|
msg.TransactionID == req.TransactionID &&
|
|
rcid != nil &&
|
|
cid.Equal(*rcid) {
|
|
log.Debug("DHCPv6: The packet is from an active DHCP server")
|
|
return true, nil
|
|
}
|
|
|
|
log.Debug("DHCPv6: received message from server doesn't match our request")
|
|
}
|
|
}
|