Pull request: dhcpd: imp normalization, validation

Updates .

Squashed commit of the following:

commit 875954fc8d59980a39b03032007cbc15d87801ea
Author: Ainar Garipov <A.Garipov@AdGuard.COM>
Date:   Wed May 5 19:54:24 2021 +0300

    all: imp err msgs

commit c6ea471038ce28f608084b59d3447ff64124260f
Author: Ainar Garipov <A.Garipov@AdGuard.COM>
Date:   Wed May 5 17:55:12 2021 +0300

    dhcpd: imp normalization, validation
This commit is contained in:
Ainar Garipov 2021-05-06 13:02:48 +03:00
parent b3cd33dde1
commit f717776c64
10 changed files with 299 additions and 273 deletions

View File

@ -17,6 +17,14 @@ and this project adheres to
## [v0.106.2] - 2021-05-17 (APPROX.) ## [v0.106.2] - 2021-05-17 (APPROX.)
--> -->
### Fixed
- Uniqueness validation for dynamic DHCP leases ([#3056]).
[#3056]: https://github.com/AdguardTeam/AdGuardHome/issues/3056
## [v0.106.1] - 2021-04-30 ## [v0.106.1] - 2021-04-30
### Fixed ### Fixed

View File

@ -48,32 +48,32 @@ const maxDomainLabelLen = 63
// See https://stackoverflow.com/a/32294443/1892060. // See https://stackoverflow.com/a/32294443/1892060.
const maxDomainNameLen = 253 const maxDomainNameLen = 253
const invalidCharMsg = "invalid char %q at index %d in %q"
// ValidateDomainNameLabel returns an error if label is not a valid label of // ValidateDomainNameLabel returns an error if label is not a valid label of
// a domain name. // a domain name.
func ValidateDomainNameLabel(label string) (err error) { func ValidateDomainNameLabel(label string) (err error) {
defer agherr.Annotate("validating label %q: %w", &err, label)
l := len(label) l := len(label)
if l > maxDomainLabelLen { if l > maxDomainLabelLen {
return fmt.Errorf("%q is too long, max: %d", label, maxDomainLabelLen) return fmt.Errorf("label is too long, max: %d", maxDomainLabelLen)
} else if l == 0 { } else if l == 0 {
return agherr.Error("label is empty") return agherr.Error("label is empty")
} }
if r := label[0]; !IsValidHostOuterRune(rune(r)) { if r := label[0]; !IsValidHostOuterRune(rune(r)) {
return fmt.Errorf(invalidCharMsg, r, 0, label) return fmt.Errorf("invalid char %q at index %d", r, 0)
} else if l == 1 { } else if l == 1 {
return nil return nil
} }
for i, r := range label[1 : l-1] { for i, r := range label[1 : l-1] {
if !isValidHostRune(r) { if !isValidHostRune(r) {
return fmt.Errorf(invalidCharMsg, r, i+1, label) return fmt.Errorf("invalid char %q at index %d", r, i+1)
} }
} }
if r := label[l-1]; !IsValidHostOuterRune(rune(r)) { if r := label[l-1]; !IsValidHostOuterRune(rune(r)) {
return fmt.Errorf(invalidCharMsg, r, l-1, label) return fmt.Errorf("invalid char %q at index %d", r, l-1)
} }
return nil return nil
@ -87,6 +87,8 @@ func ValidateDomainNameLabel(label string) (err error) {
// TODO(a.garipov): After making sure that this works correctly, port this into // TODO(a.garipov): After making sure that this works correctly, port this into
// module golibs. // module golibs.
func ValidateDomainName(name string) (err error) { func ValidateDomainName(name string) (err error) {
defer agherr.Annotate("validating domain name %q: %w", &err, name)
name, err = idna.ToASCII(name) name, err = idna.ToASCII(name)
if err != nil { if err != nil {
return err return err
@ -96,7 +98,7 @@ func ValidateDomainName(name string) (err error) {
if l == 0 { if l == 0 {
return agherr.Error("domain name is empty") return agherr.Error("domain name is empty")
} else if l > maxDomainNameLen { } else if l > maxDomainNameLen {
return fmt.Errorf("%q is too long, max: %d", name, maxDomainNameLen) return fmt.Errorf("too long, max: %d", maxDomainNameLen)
} }
labels := strings.Split(name, ".") labels := strings.Split(name, ".")

View File

@ -95,40 +95,46 @@ func TestValidateDomainName(t *testing.T) {
}, { }, {
name: "empty", name: "empty",
in: "", in: "",
wantErrMsg: `domain name is empty`, wantErrMsg: `validating domain name "": domain name is empty`,
}, { }, {
name: "bad_symbol", name: "bad_symbol",
in: "!!!", in: "!!!",
wantErrMsg: `invalid domain name label at index 0: ` + wantErrMsg: `validating domain name "!!!": invalid domain name label at index 0: ` +
`invalid char '!' at index 0 in "!!!"`, `validating label "!!!": invalid char '!' at index 0`,
}, { }, {
name: "bad_length", name: "bad_length",
in: longDomainName, in: longDomainName,
wantErrMsg: `"` + longDomainName + `" is too long, max: 253`, wantErrMsg: `validating domain name "` + longDomainName + `": too long, max: 253`,
}, { }, {
name: "bad_label_length", name: "bad_label_length",
in: longLabelDomainName, in: longLabelDomainName,
wantErrMsg: `invalid domain name label at index 0: "` + longLabel + wantErrMsg: `validating domain name "` + longLabelDomainName + `": ` +
`" is too long, max: 63`, `invalid domain name label at index 0: validating label "` + longLabel +
`": label is too long, max: 63`,
}, { }, {
name: "bad_label_empty", name: "bad_label_empty",
in: "example..com", in: "example..com",
wantErrMsg: `invalid domain name label at index 1: label is empty`, wantErrMsg: `validating domain name "example..com": ` +
`invalid domain name label at index 1: ` +
`validating label "": label is empty`,
}, { }, {
name: "bad_label_first_symbol", name: "bad_label_first_symbol",
in: "example.-aa.com", in: "example.-aa.com",
wantErrMsg: `invalid domain name label at index 1:` + wantErrMsg: `validating domain name "example.-aa.com": ` +
` invalid char '-' at index 0 in "-aa"`, `invalid domain name label at index 1: ` +
`validating label "-aa": invalid char '-' at index 0`,
}, { }, {
name: "bad_label_last_symbol", name: "bad_label_last_symbol",
in: "example-.aa.com", in: "example-.aa.com",
wantErrMsg: `invalid domain name label at index 0:` + wantErrMsg: `validating domain name "example-.aa.com": ` +
` invalid char '-' at index 7 in "example-"`, `invalid domain name label at index 0: ` +
`validating label "example-": invalid char '-' at index 7`,
}, { }, {
name: "bad_label_symbol", name: "bad_label_symbol",
in: "example.a!!!.com", in: "example.a!!!.com",
wantErrMsg: `invalid domain name label at index 1:` + wantErrMsg: `validating domain name "example.a!!!.com": ` +
` invalid char '!' at index 1 in "a!!!"`, `invalid domain name label at index 1: ` +
`validating label "a!!!": invalid char '!' at index 1`,
}} }}
for _, tc := range testCases { for _, tc := range testCases {

View File

@ -27,13 +27,13 @@ var webHandlersRegistered = false
// Lease contains the necessary information about a DHCP lease // Lease contains the necessary information about a DHCP lease
type Lease struct { type Lease struct {
// Expiry is the expiration time of the lease. The unix timestamp value
// of 1 means that this is a static lease.
Expiry time.Time `json:"expires"`
Hostname string `json:"hostname"`
HWAddr net.HardwareAddr `json:"mac"` HWAddr net.HardwareAddr `json:"mac"`
IP net.IP `json:"ip"` IP net.IP `json:"ip"`
Hostname string `json:"hostname"`
// Lease expiration time
// 1: static lease
Expiry time.Time `json:"expires"`
} }
// IsStatic returns true if the lease is static. // IsStatic returns true if the lease is static.

View File

@ -37,30 +37,34 @@ func TestDB(t *testing.T) {
SubnetMask: net.IP{255, 255, 255, 0}, SubnetMask: net.IP{255, 255, 255, 0},
notify: testNotify, notify: testNotify,
}) })
require.Nil(t, err) require.NoError(t, err)
s.srv6, err = v6Create(V6ServerConf{}) s.srv6, err = v6Create(V6ServerConf{})
require.Nil(t, err) require.NoError(t, err)
leases := []Lease{{ leases := []Lease{{
IP: net.IP{192, 168, 10, 100}, Expiry: time.Now().Add(time.Hour),
HWAddr: net.HardwareAddr{0xAA, 0xAA, 0xAA, 0xAA, 0xAA, 0xAA}, Hostname: "static-1.local",
Expiry: time.Now().Add(time.Hour), HWAddr: net.HardwareAddr{0xAA, 0xAA, 0xAA, 0xAA, 0xAA, 0xAA},
IP: net.IP{192, 168, 10, 100},
}, { }, {
IP: net.IP{192, 168, 10, 101}, Hostname: "static-2.local",
HWAddr: net.HardwareAddr{0xAA, 0xAA, 0xAA, 0xAA, 0xAA, 0xBB}, HWAddr: net.HardwareAddr{0xAA, 0xAA, 0xAA, 0xAA, 0xAA, 0xBB},
IP: net.IP{192, 168, 10, 101},
}} }}
srv4, ok := s.srv4.(*v4Server) srv4, ok := s.srv4.(*v4Server)
require.True(t, ok) require.True(t, ok)
err = srv4.addLease(&leases[0]) err = srv4.addLease(&leases[0])
require.Nil(t, err) require.NoError(t, err)
require.Nil(t, s.srv4.AddStaticLease(leases[1]))
err = s.srv4.AddStaticLease(leases[1])
require.NoError(t, err)
s.dbStore() s.dbStore()
t.Cleanup(func() { t.Cleanup(func() {
assert.Nil(t, os.Remove(dbFilename)) assert.NoError(t, os.Remove(dbFilename))
}) })
s.srv4.ResetLeases(nil) s.srv4.ResetLeases(nil)
s.dbLoad() s.dbLoad()

View File

@ -36,7 +36,7 @@ type v4Server struct {
// leases contains all dynamic and static leases. // leases contains all dynamic and static leases.
leases []*Lease leases []*Lease
// leasesLock protects leases and leasedOffsets. // leasesLock protects leases, leaseHosts, and leasedOffsets.
leasesLock sync.Mutex leasesLock sync.Mutex
} }
@ -49,6 +49,68 @@ func (s *v4Server) WriteDiskConfig4(c *V4ServerConf) {
func (s *v4Server) WriteDiskConfig6(c *V6ServerConf) { func (s *v4Server) WriteDiskConfig6(c *V6ServerConf) {
} }
// normalizeHostname normalizes a hostname sent by the client. If err is not
// nil, norm is an empty string.
func normalizeHostname(hostname string) (norm string, err error) {
defer agherr.Annotate("normalizing %q: %w", &err, hostname)
if hostname == "" {
return "", nil
}
norm = strings.ToLower(hostname)
parts := strings.FieldsFunc(norm, func(c rune) (ok bool) {
return c != '.' && !aghnet.IsValidHostOuterRune(c)
})
if len(parts) == 0 {
return "", fmt.Errorf("no valid parts")
}
norm = strings.Join(parts, "-")
norm = strings.TrimSuffix(norm, "-")
return norm, nil
}
// validHostnameForClient accepts the hostname sent by the client and returns
// either a normalized version of that hostname or a new hostname generated from
// the client's IP address. If this new hostname is different from the provided
// previous hostname, additional uniqueness check is performed.
//
// hostname is always a non-empty valid hostname. If err is not nil, it
// describes the issues encountered when normalizing cliHostname.
func (s *v4Server) validHostnameForClient(
cliHostname string,
prevHostname string,
ip net.IP,
) (hostname string, err error) {
hostname, err = normalizeHostname(cliHostname)
if err == nil && hostname != "" {
err = aghnet.ValidateDomainName(hostname)
if err != nil {
// Go on and assign a hostname made from the IP below,
// returning the error that we've got.
hostname = ""
} else if hostname != prevHostname && s.leaseHosts.Has(hostname) {
// Go on and assign a unique hostname made from the IP
// below, returning the error about uniqueness.
err = agherr.Error("hostname exists")
hostname = ""
}
}
if hostname == "" {
hostname = aghnet.GenerateHostname(ip)
}
if hostname != cliHostname {
log.Info("dhcpv4: normalized hostname %q into %q", cliHostname, hostname)
}
return hostname, err
}
// ResetLeases - reset leases // ResetLeases - reset leases
func (s *v4Server) ResetLeases(leases []*Lease) { func (s *v4Server) ResetLeases(leases []*Lease) {
var err error var err error
@ -62,9 +124,13 @@ func (s *v4Server) ResetLeases(leases []*Lease) {
s.leases = nil s.leases = nil
for _, l := range leases { for _, l := range leases {
l.Hostname, err = s.validHostnameForClient(l.Hostname, l.IP) l.Hostname, err = s.validHostnameForClient(l.Hostname, l.Hostname, l.IP)
if err != nil { if err != nil {
log.Info("dhcpv4: warning: previous hostname %q is invalid: %s", l.Hostname, err) log.Info(
"dhcpv4: warning: previous hostname %q is invalid: %s",
l.Hostname,
err,
)
} }
err = s.addLease(l) err = s.addLease(l)
@ -204,7 +270,7 @@ func (s *v4Server) rmDynamicLease(lease *Lease) (err error) {
if bytes.Equal(l.HWAddr, lease.HWAddr) { if bytes.Equal(l.HWAddr, lease.HWAddr) {
if l.IsStatic() { if l.IsStatic() {
return fmt.Errorf("static lease already exists") return agherr.Error("static lease already exists")
} }
s.rmLeaseByIndex(i) s.rmLeaseByIndex(i)
@ -215,9 +281,9 @@ func (s *v4Server) rmDynamicLease(lease *Lease) (err error) {
l = s.leases[i] l = s.leases[i]
} }
if net.IP.Equal(l.IP, lease.IP) { if l.IP.Equal(lease.IP) {
if l.IsStatic() { if l.IsStatic() {
return fmt.Errorf("static lease already exists") return agherr.Error("static lease already exists")
} }
s.rmLeaseByIndex(i) s.rmLeaseByIndex(i)
@ -227,54 +293,31 @@ func (s *v4Server) rmDynamicLease(lease *Lease) (err error) {
return nil return nil
} }
func (s *v4Server) addStaticLease(l *Lease) (err error) { // addLease adds a dynamic or static lease.
if sn := s.conf.subnet; !sn.Contains(l.IP) { func (s *v4Server) addLease(l *Lease) (err error) {
return fmt.Errorf("subnet %s does not contain the ip %q", sn, l.IP)
}
s.leases = append(s.leases, l)
r := s.conf.ipRange r := s.conf.ipRange
offset, ok := r.offset(l.IP) offset, inOffset := r.offset(l.IP)
if ok {
s.leasedOffsets.set(offset, true)
}
s.leaseHosts.Add(l.Hostname) if l.IsStatic() {
if sn := s.conf.subnet; !sn.Contains(l.IP) {
return nil return fmt.Errorf("subnet %s does not contain the ip %q", sn, l.IP)
} }
} else if !inOffset {
func (s *v4Server) addDynamicLease(l *Lease) (err error) {
r := s.conf.ipRange
offset, ok := r.offset(l.IP)
if !ok {
return fmt.Errorf("lease %s (%s) out of range, not adding", l.IP, l.HWAddr) return fmt.Errorf("lease %s (%s) out of range, not adding", l.IP, l.HWAddr)
} }
s.leases = append(s.leases, l) s.leases = append(s.leases, l)
s.leaseHosts.Add(l.Hostname)
s.leasedOffsets.set(offset, true) s.leasedOffsets.set(offset, true)
if l.Hostname != "" {
s.leaseHosts.Add(l.Hostname)
}
return nil return nil
} }
// addLease adds a dynamic or static lease. // rmLease removes a lease with the same properties.
func (s *v4Server) addLease(l *Lease) (err error) { func (s *v4Server) rmLease(lease Lease) (err error) {
err = s.validateLease(l)
if err != nil {
return err
}
if l.IsStatic() {
return s.addStaticLease(l)
}
return s.addDynamicLease(l)
}
// Remove a lease with the same properties
func (s *v4Server) rmLease(lease Lease) error {
if len(s.leases) == 0 { if len(s.leases) == 0 {
return nil return nil
} }
@ -296,7 +339,7 @@ func (s *v4Server) rmLease(lease Lease) error {
// AddStaticLease adds a static lease. It is safe for concurrent use. // AddStaticLease adds a static lease. It is safe for concurrent use.
func (s *v4Server) AddStaticLease(l Lease) (err error) { func (s *v4Server) AddStaticLease(l Lease) (err error) {
defer agherr.Annotate("dhcpv4: %w", &err) defer agherr.Annotate("dhcpv4: adding static lease: %w", &err)
if ip4 := l.IP.To4(); ip4 == nil { if ip4 := l.IP.To4(); ip4 == nil {
return fmt.Errorf("invalid ip %q, only ipv4 is supported", l.IP) return fmt.Errorf("invalid ip %q, only ipv4 is supported", l.IP)
@ -304,11 +347,28 @@ func (s *v4Server) AddStaticLease(l Lease) (err error) {
l.Expiry = time.Unix(leaseExpireStatic, 0) l.Expiry = time.Unix(leaseExpireStatic, 0)
l.Hostname, err = normalizeHostname(l.Hostname) err = aghnet.ValidateHardwareAddress(l.HWAddr)
if err != nil { if err != nil {
return err return err
} }
var hostname string
hostname, err = normalizeHostname(l.Hostname)
if err != nil {
return err
}
err = aghnet.ValidateDomainName(hostname)
if err != nil {
return fmt.Errorf("validating hostname: %w", err)
}
if s.leaseHosts.Has(hostname) {
return agherr.Error("hostname exists")
}
l.Hostname = hostname
// Perform the following actions in an anonymous function to make sure // Perform the following actions in an anonymous function to make sure
// that the lock gets unlocked before the notification step. // that the lock gets unlocked before the notification step.
func() { func() {
@ -372,16 +432,19 @@ func (s *v4Server) RemoveStaticLease(l Lease) (err error) {
return nil return nil
} }
// Send ICMP to the specified machine // addrAvailable sends an ICP request to the specified IP address. It returns
// Return TRUE if it doesn't reply, which probably means that the IP is available // true if the remote host doesn't reply, which probably means that the IP
func (s *v4Server) addrAvailable(target net.IP) bool { // address is available.
//
// TODO(a.garipov): I'm not sure that this is the best way to do this.
func (s *v4Server) addrAvailable(target net.IP) (avail bool) {
if s.conf.ICMPTimeout == 0 { if s.conf.ICMPTimeout == 0 {
return true return true
} }
pinger, err := ping.NewPinger(target.String()) pinger, err := ping.NewPinger(target.String())
if err != nil { if err != nil {
log.Error("ping.NewPinger(): %v", err) log.Error("dhcpv4: ping.NewPinger(): %s", err)
return true return true
} }
@ -393,20 +456,24 @@ func (s *v4Server) addrAvailable(target net.IP) bool {
pinger.OnRecv = func(_ *ping.Packet) { pinger.OnRecv = func(_ *ping.Packet) {
reply = true reply = true
} }
log.Debug("dhcpv4: Sending ICMP Echo to %v", target)
log.Debug("dhcpv4: sending icmp echo to %s", target)
err = pinger.Run() err = pinger.Run()
if err != nil { if err != nil {
log.Error("pinger.Run(): %v", err) log.Error("dhcpv4: pinger.Run(): %s", err)
return true return true
} }
if reply { if reply {
log.Info("dhcpv4: IP conflict: %v is already used by another device", target) log.Info("dhcpv4: ip conflict: %s is already used by another device", target)
return false return false
} }
log.Debug("dhcpv4: ICMP procedure is complete: %v", target) log.Debug("dhcpv4: icmp procedure is complete: %q", target)
return true return true
} }
@ -481,58 +548,69 @@ func (s *v4Server) reserveLease(mac net.HardwareAddr) (l *Lease, err error) {
func (s *v4Server) commitLease(l *Lease) { func (s *v4Server) commitLease(l *Lease) {
l.Expiry = time.Now().Add(s.conf.leaseTime) l.Expiry = time.Now().Add(s.conf.leaseTime)
s.leasesLock.Lock() func() {
s.conf.notify(LeaseChangedDBStore) s.leasesLock.Lock()
s.leasesLock.Unlock() defer s.leasesLock.Unlock()
s.conf.notify(LeaseChangedDBStore)
s.leaseHosts.Add(l.Hostname)
}()
s.conf.notify(LeaseChangedAdded) s.conf.notify(LeaseChangedAdded)
} }
// Process Discover request and return lease // processDiscover is the handler for the DHCP Discover request.
func (s *v4Server) processDiscover(req, resp *dhcpv4.DHCPv4) (l *Lease, err error) { func (s *v4Server) processDiscover(req, resp *dhcpv4.DHCPv4) (l *Lease, err error) {
mac := req.ClientHWAddr mac := req.ClientHWAddr
err = aghnet.ValidateHardwareAddress(mac)
if err != nil {
return nil, err
}
s.leasesLock.Lock() s.leasesLock.Lock()
defer s.leasesLock.Unlock() defer s.leasesLock.Unlock()
// TODO(a.garipov): Refactor this mess.
l = s.findLease(mac) l = s.findLease(mac)
if l == nil { if l != nil {
toStore := false
for l == nil {
l, err = s.reserveLease(mac)
if err != nil {
return nil, fmt.Errorf("reserving a lease: %w", err)
}
if l == nil {
log.Debug("dhcpv4: no more ip addresses")
if toStore {
s.conf.notify(LeaseChangedDBStore)
}
// TODO(a.garipov): Return a special error?
return nil, nil
}
toStore = true
if !s.addrAvailable(l.IP) {
s.blocklistLease(l)
l = nil
continue
}
break
}
s.conf.notify(LeaseChangedDBStore)
} else {
reqIP := req.RequestedIPAddress() reqIP := req.RequestedIPAddress()
if len(reqIP) != 0 && !reqIP.Equal(l.IP) { if len(reqIP) != 0 && !reqIP.Equal(l.IP) {
log.Debug("dhcpv4: different RequestedIP: %s != %s", reqIP, l.IP) log.Debug("dhcpv4: different RequestedIP: %s != %s", reqIP, l.IP)
} }
resp.UpdateOption(dhcpv4.OptMessageType(dhcpv4.MessageTypeOffer))
return l, nil
}
needsUpdate := false
defer func() {
if needsUpdate {
s.conf.notify(LeaseChangedDBStore)
}
}()
leaseReady := false
for !leaseReady {
l, err = s.reserveLease(mac)
if err != nil {
return nil, fmt.Errorf("reserving a lease: %w", err)
}
if l == nil {
log.Debug("dhcpv4: no more ip addresses")
return nil, nil
}
needsUpdate = true
if s.addrAvailable(l.IP) {
leaseReady = true
} else {
s.blocklistLease(l)
l = nil
}
} }
resp.UpdateOption(dhcpv4.OptMessageType(dhcpv4.MessageTypeOffer)) resp.UpdateOption(dhcpv4.OptMessageType(dhcpv4.MessageTypeOffer))
@ -569,115 +647,14 @@ func (o *optFQDN) ToBytes() []byte {
return b return b
} }
// normalizeHostname normalizes a hostname sent by the client. If err is not // processDiscover is the handler for the DHCP Request request.
// nil, norm is an empty string. func (s *v4Server) processRequest(req, resp *dhcpv4.DHCPv4) (lease *Lease, needsReply bool) {
func normalizeHostname(name string) (norm string, err error) {
if name == "" {
return "", nil
}
norm = strings.ToLower(name)
parts := strings.FieldsFunc(norm, func(c rune) (ok bool) {
return c != '.' && !aghnet.IsValidHostOuterRune(c)
})
if len(parts) == 0 {
return "", fmt.Errorf("normalizing hostname %q: no valid parts", name)
}
norm = strings.Join(parts, "-")
norm = strings.TrimSuffix(norm, "-")
return norm, nil
}
// validateHostname validates a hostname sent by the client.
func (s *v4Server) validateHostname(name string) (err error) {
defer agherr.Annotate("validating hostname: %s", &err)
if name == "" {
return nil
}
err = aghnet.ValidateDomainName(name)
if err != nil {
return err
}
if s.leaseHosts.Has(name) {
return agherr.Error("hostname exists")
}
return nil
}
// validHostnameForClient accepts the hostname sent by the client and returns
// either a normalized version of that hostname or a new hostname generated from
// the client's IP address.
//
// hostname is always a non-empty valid hostname. If err is not nil, it
// describes the issues encountered when normalizing cliHostname.
func (s *v4Server) validHostnameForClient(
cliHostname string,
ip net.IP,
) (hostname string, err error) {
hostname, err = normalizeHostname(cliHostname)
if err == nil {
err = s.validateHostname(hostname)
if err != nil {
// Go on and assign a hostname made from the IP below,
// returning the error that we've got.
hostname = ""
}
}
if hostname == "" {
hostname = aghnet.GenerateHostname(ip)
}
if hostname != cliHostname {
log.Info("dhcpv4: normalized hostname %q into %q", cliHostname, hostname)
}
return hostname, err
}
// validateLease returns an error if the lease is invalid.
func (s *v4Server) validateLease(l *Lease) (err error) {
defer agherr.Annotate("validating lease: %s", &err)
if l == nil {
return agherr.Error("lease is nil")
}
err = aghnet.ValidateHardwareAddress(l.HWAddr)
if err != nil {
return err
}
err = s.validateHostname(l.Hostname)
if err != nil {
return err
}
if sn := s.conf.subnet; !sn.Contains(l.IP) {
return fmt.Errorf("subnet %s does not contain the ip %q", sn, l.IP)
}
r := s.conf.ipRange
if !l.IsStatic() && !r.contains(l.IP) {
return fmt.Errorf("dynamic lease range %s does not contain the ip %q", r, l.IP)
}
return nil
}
// Process Request request and return lease
// Return false if we don't need to reply
func (s *v4Server) processRequest(req, resp *dhcpv4.DHCPv4) (lease *Lease, ok bool) {
var err error
mac := req.ClientHWAddr mac := req.ClientHWAddr
err := aghnet.ValidateHardwareAddress(mac)
if err != nil {
return nil, false
}
reqIP := req.RequestedIPAddress() reqIP := req.RequestedIPAddress()
if reqIP == nil { if reqIP == nil {
reqIP = req.ClientIPAddr reqIP = req.ClientIPAddr
@ -696,34 +673,49 @@ func (s *v4Server) processRequest(req, resp *dhcpv4.DHCPv4) (lease *Lease, ok bo
return nil, false return nil, false
} }
s.leasesLock.Lock() mismatch := false
for _, l := range s.leases { func() {
if bytes.Equal(l.HWAddr, mac) { s.leasesLock.Lock()
if !l.IP.Equal(reqIP) { defer s.leasesLock.Unlock()
s.leasesLock.Unlock()
log.Debug("dhcpv4: mismatched OptionRequestedIPAddress in request message for %s", mac)
return nil, true for _, l := range s.leases {
if !bytes.Equal(l.HWAddr, mac) {
continue
} }
lease = l if l.IP.Equal(reqIP) {
lease = l
} else {
log.Debug(
`dhcpv4: mismatched OptionRequestedIPAddress `+
`in request message for %s`,
mac,
)
mismatch = true
}
break return
} }
}()
if mismatch {
return nil, true
} }
s.leasesLock.Unlock()
if lease == nil { if lease == nil {
log.Debug("dhcpv4: no lease for %s", mac) log.Debug("dhcpv4: no reserved lease for %s", mac)
return nil, true return nil, true
} }
if !lease.IsStatic() { if !lease.IsStatic() {
cliHostname := req.HostName() cliHostname := req.HostName()
lease.Hostname, err = s.validHostnameForClient(cliHostname, reqIP) lease.Hostname, err = s.validHostnameForClient(cliHostname, lease.Hostname, reqIP)
if err != nil { if err != nil {
log.Info("dhcpv4: warning: client hostname %q is invalid: %s", cliHostname, err) log.Info(
"dhcpv4: warning: client hostname %q is invalid: %s",
cliHostname,
err,
)
} }
s.commitLease(lease) s.commitLease(lease)

View File

@ -30,8 +30,9 @@ func TestV4_AddRemove_static(t *testing.T) {
// Add static lease. // Add static lease.
l := Lease{ l := Lease{
IP: net.IP{192, 168, 10, 150}, Hostname: "static-1.local",
HWAddr: net.HardwareAddr{0xAA, 0xAA, 0xAA, 0xAA, 0xAA, 0xAA}, HWAddr: net.HardwareAddr{0xAA, 0xAA, 0xAA, 0xAA, 0xAA, 0xAA},
IP: net.IP{192, 168, 10, 150},
} }
err = s.AddStaticLease(l) err = s.AddStaticLease(l)
@ -76,11 +77,13 @@ func TestV4_AddReplace(t *testing.T) {
require.True(t, ok) require.True(t, ok)
dynLeases := []Lease{{ dynLeases := []Lease{{
IP: net.IP{192, 168, 10, 150}, Hostname: "dynamic-1.local",
HWAddr: net.HardwareAddr{0x11, 0xAA, 0xAA, 0xAA, 0xAA, 0xAA}, HWAddr: net.HardwareAddr{0x11, 0xAA, 0xAA, 0xAA, 0xAA, 0xAA},
IP: net.IP{192, 168, 10, 150},
}, { }, {
IP: net.IP{192, 168, 10, 151}, Hostname: "dynamic-2.local",
HWAddr: net.HardwareAddr{0x22, 0xAA, 0xAA, 0xAA, 0xAA, 0xAA}, HWAddr: net.HardwareAddr{0x22, 0xAA, 0xAA, 0xAA, 0xAA, 0xAA},
IP: net.IP{192, 168, 10, 151},
}} }}
for i := range dynLeases { for i := range dynLeases {
@ -89,11 +92,13 @@ func TestV4_AddReplace(t *testing.T) {
} }
stLeases := []Lease{{ stLeases := []Lease{{
IP: net.IP{192, 168, 10, 150}, Hostname: "static-1.local",
HWAddr: net.HardwareAddr{0x33, 0xAA, 0xAA, 0xAA, 0xAA, 0xAA}, HWAddr: net.HardwareAddr{0x33, 0xAA, 0xAA, 0xAA, 0xAA, 0xAA},
IP: net.IP{192, 168, 10, 150},
}, { }, {
IP: net.IP{192, 168, 10, 152}, Hostname: "static-2.local",
HWAddr: net.HardwareAddr{0x22, 0xAA, 0xAA, 0xAA, 0xAA, 0xAA}, HWAddr: net.HardwareAddr{0x22, 0xAA, 0xAA, 0xAA, 0xAA, 0xAA},
IP: net.IP{192, 168, 10, 152},
}} }}
for _, l := range stLeases { for _, l := range stLeases {
@ -129,8 +134,9 @@ func TestV4StaticLease_Get(t *testing.T) {
s.conf.dnsIPAddrs = []net.IP{{192, 168, 10, 1}} s.conf.dnsIPAddrs = []net.IP{{192, 168, 10, 1}}
l := Lease{ l := Lease{
IP: net.IP{192, 168, 10, 150}, Hostname: "static-1.local",
HWAddr: net.HardwareAddr{0xAA, 0xAA, 0xAA, 0xAA, 0xAA, 0xAA}, HWAddr: net.HardwareAddr{0xAA, 0xAA, 0xAA, 0xAA, 0xAA, 0xAA},
IP: net.IP{192, 168, 10, 150},
} }
err = s.AddStaticLease(l) err = s.AddStaticLease(l)
require.NoError(t, err) require.NoError(t, err)
@ -269,7 +275,12 @@ func TestV4DynamicLease_Get(t *testing.T) {
assert.Equal(t, dhcpv4.MessageTypeAck, resp.MessageType()) assert.Equal(t, dhcpv4.MessageTypeAck, resp.MessageType())
assert.Equal(t, mac, resp.ClientHWAddr) assert.Equal(t, mac, resp.ClientHWAddr)
assert.True(t, s.conf.RangeStart.Equal(resp.YourIPAddr)) assert.True(t, s.conf.RangeStart.Equal(resp.YourIPAddr))
assert.True(t, s.conf.GatewayIP.Equal(resp.Router()[0]))
router := resp.Router()
require.Len(t, router, 1)
assert.Equal(t, s.conf.GatewayIP, router[0])
assert.True(t, s.conf.GatewayIP.Equal(resp.ServerIdentifier())) assert.True(t, s.conf.GatewayIP.Equal(resp.ServerIdentifier()))
assert.Equal(t, s.conf.subnet.Mask, resp.SubnetMask()) assert.Equal(t, s.conf.subnet.Mask, resp.SubnetMask())
assert.Equal(t, s.conf.leaseTime.Seconds(), resp.IPAddressLeaseTime(-1).Seconds()) assert.Equal(t, s.conf.leaseTime.Seconds(), resp.IPAddressLeaseTime(-1).Seconds())
@ -329,12 +340,12 @@ func TestNormalizeHostname(t *testing.T) {
}, { }, {
name: "error", name: "error",
hostname: "!!!", hostname: "!!!",
wantErrMsg: `normalizing hostname "!!!": no valid parts`, wantErrMsg: `normalizing "!!!": no valid parts`,
want: "", want: "",
}, { }, {
name: "error_spaces", name: "error_spaces",
hostname: "! ! !", hostname: "! ! !",
wantErrMsg: `normalizing hostname "! ! !": no valid parts`, wantErrMsg: `normalizing "! ! !": no valid parts`,
want: "", want: "",
}} }}

View File

@ -2,6 +2,7 @@ package dnsforward
import ( import (
"crypto/tls" "crypto/tls"
"errors"
"fmt" "fmt"
"path" "path"
"strings" "strings"
@ -15,7 +16,8 @@ import (
func ValidateClientID(clientID string) (err error) { func ValidateClientID(clientID string) (err error) {
err = aghnet.ValidateDomainNameLabel(clientID) err = aghnet.ValidateDomainNameLabel(clientID)
if err != nil { if err != nil {
return fmt.Errorf("invalid client id: %w", err) // Replace the domain name label wrapper with our own.
return fmt.Errorf("invalid client id %q: %w", clientID, errors.Unwrap(err))
} }
return nil return nil

View File

@ -117,8 +117,8 @@ func TestProcessClientID(t *testing.T) {
hostSrvName: "example.com", hostSrvName: "example.com",
cliSrvName: "!!!.example.com", cliSrvName: "!!!.example.com",
wantClientID: "", wantClientID: "",
wantErrMsg: `client id check: invalid client id: invalid char '!' ` + wantErrMsg: `client id check: invalid client id "!!!": ` +
`at index 0 in "!!!"`, `invalid char '!' at index 0`,
wantRes: resultCodeError, wantRes: resultCodeError,
strictSNI: true, strictSNI: true,
}, { }, {
@ -128,9 +128,9 @@ func TestProcessClientID(t *testing.T) {
cliSrvName: `abcdefghijklmnopqrstuvwxyz0123456789abcdefghijklmno` + cliSrvName: `abcdefghijklmnopqrstuvwxyz0123456789abcdefghijklmno` +
`pqrstuvwxyz0123456789.example.com`, `pqrstuvwxyz0123456789.example.com`,
wantClientID: "", wantClientID: "",
wantErrMsg: `client id check: invalid client id: "abcdefghijklmno` + wantErrMsg: `client id check: invalid client id "abcdefghijklmno` +
`pqrstuvwxyz0123456789abcdefghijklmnopqrstuvwxyz0123456789" ` + `pqrstuvwxyz0123456789abcdefghijklmnopqrstuvwxyz0123456789": ` +
`is too long, max: 63`, `label is too long, max: 63`,
wantRes: resultCodeError, wantRes: resultCodeError,
strictSNI: true, strictSNI: true,
}, { }, {
@ -238,8 +238,8 @@ func TestProcessClientID_https(t *testing.T) {
name: "invalid_client_id", name: "invalid_client_id",
path: "/dns-query/!!!", path: "/dns-query/!!!",
wantClientID: "", wantClientID: "",
wantErrMsg: `client id check: invalid client id: invalid char '!' ` + wantErrMsg: `client id check: invalid client id "!!!": ` +
`at index 0 in "!!!"`, `invalid char '!' at index 0`,
wantRes: resultCodeError, wantRes: resultCodeError,
}} }}

View File

@ -1166,8 +1166,9 @@ func TestNewServer(t *testing.T) {
in: DNSCreateParams{ in: DNSCreateParams{
LocalDomain: "!!!", LocalDomain: "!!!",
}, },
wantErrMsg: `local domain: invalid domain name label at index 0: ` + wantErrMsg: `local domain: validating domain name "!!!": ` +
`invalid char '!' at index 0 in "!!!"`, `invalid domain name label at index 0: ` +
`validating label "!!!": invalid char '!' at index 0`,
}} }}
for _, tc := range testCases { for _, tc := range testCases {