mirror of
https://github.com/AdguardTeam/AdGuardHome.git
synced 2024-12-20 22:11:50 +03:00
beeb8f0522
Updates #4923. Squashed commit of the following: commit 0bfccf8bc1e63c4f5a01ce7f268e767969b368a0 Merge:305f9fe2f
0e5e8e4dd
Author: Eugene Burkov <E.Burkov@AdGuard.COM> Date: Wed Jul 3 15:12:43 2024 +0300 Merge branch 'master' into 4923-gopacket-dhcp-vol.8 commit305f9fe2fe
Author: Eugene Burkov <E.Burkov@AdGuard.COM> Date: Tue Jul 2 17:03:01 2024 +0300 dhcpsvc: adjust interface commitf05b9f42e2
Author: Eugene Burkov <E.Burkov@AdGuard.COM> Date: Tue Jul 2 16:59:39 2024 +0300 dhcpsvc: use logger commit4779f945ba
Author: Eugene Burkov <E.Burkov@AdGuard.COM> Date: Tue Jul 2 14:59:22 2024 +0300 dhcpsvc: add todo commitae1713e5f7
Author: Eugene Burkov <E.Burkov@AdGuard.COM> Date: Mon Jul 1 15:49:22 2024 +0300 dhcpsvc: use slog
295 lines
7 KiB
Go
295 lines
7 KiB
Go
package dhcpsvc
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"log/slog"
|
|
"net"
|
|
"net/netip"
|
|
"sync"
|
|
"sync/atomic"
|
|
"time"
|
|
|
|
"github.com/AdguardTeam/golibs/errors"
|
|
"github.com/AdguardTeam/golibs/mapsutil"
|
|
)
|
|
|
|
// DHCPServer is a DHCP server for both IPv4 and IPv6 address families.
|
|
type DHCPServer struct {
|
|
// enabled indicates whether the DHCP server is enabled and can provide
|
|
// information about its clients.
|
|
enabled *atomic.Bool
|
|
|
|
// logger logs common DHCP events.
|
|
logger *slog.Logger
|
|
|
|
// localTLD is the top-level domain name to use for resolving DHCP clients'
|
|
// hostnames.
|
|
localTLD string
|
|
|
|
// leasesMu protects the leases index as well as leases in the interfaces.
|
|
leasesMu *sync.RWMutex
|
|
|
|
// leases stores the DHCP leases for quick lookups.
|
|
leases *leaseIndex
|
|
|
|
// interfaces4 is the set of IPv4 interfaces sorted by interface name.
|
|
interfaces4 netInterfacesV4
|
|
|
|
// interfaces6 is the set of IPv6 interfaces sorted by interface name.
|
|
interfaces6 netInterfacesV6
|
|
|
|
// icmpTimeout is the timeout for checking another DHCP server's presence.
|
|
icmpTimeout time.Duration
|
|
}
|
|
|
|
// New creates a new DHCP server with the given configuration. It returns an
|
|
// error if the given configuration can't be used.
|
|
//
|
|
// TODO(e.burkov): Use.
|
|
func New(ctx context.Context, conf *Config) (srv *DHCPServer, err error) {
|
|
l := conf.Logger
|
|
if !conf.Enabled {
|
|
l.DebugContext(ctx, "disabled")
|
|
|
|
// TODO(e.burkov): Perhaps return [Empty]?
|
|
return nil, nil
|
|
}
|
|
|
|
// TODO(e.burkov): Add validations scoped to the network interfaces set.
|
|
ifaces4 := make(netInterfacesV4, 0, len(conf.Interfaces))
|
|
ifaces6 := make(netInterfacesV6, 0, len(conf.Interfaces))
|
|
var errs []error
|
|
|
|
mapsutil.SortedRange(conf.Interfaces, func(name string, iface *InterfaceConfig) (cont bool) {
|
|
var i4 *netInterfaceV4
|
|
i4, err = newNetInterfaceV4(ctx, l, name, iface.IPv4)
|
|
if err != nil {
|
|
errs = append(errs, fmt.Errorf("interface %q: ipv4: %w", name, err))
|
|
} else if i4 != nil {
|
|
ifaces4 = append(ifaces4, i4)
|
|
}
|
|
|
|
i6 := newNetInterfaceV6(ctx, l, name, iface.IPv6)
|
|
if i6 != nil {
|
|
ifaces6 = append(ifaces6, i6)
|
|
}
|
|
|
|
return true
|
|
})
|
|
if err = errors.Join(errs...); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
enabled := &atomic.Bool{}
|
|
enabled.Store(conf.Enabled)
|
|
|
|
srv = &DHCPServer{
|
|
enabled: enabled,
|
|
logger: l,
|
|
localTLD: conf.LocalDomainName,
|
|
leasesMu: &sync.RWMutex{},
|
|
leases: newLeaseIndex(),
|
|
interfaces4: ifaces4,
|
|
interfaces6: ifaces6,
|
|
icmpTimeout: conf.ICMPTimeout,
|
|
}
|
|
|
|
// TODO(e.burkov): Load leases.
|
|
|
|
return srv, nil
|
|
}
|
|
|
|
// type check
|
|
//
|
|
// TODO(e.burkov): Uncomment when the [Interface] interface is implemented.
|
|
// var _ Interface = (*DHCPServer)(nil)
|
|
|
|
// Enabled implements the [Interface] interface for *DHCPServer.
|
|
func (srv *DHCPServer) Enabled() (ok bool) {
|
|
return srv.enabled.Load()
|
|
}
|
|
|
|
// Leases implements the [Interface] interface for *DHCPServer.
|
|
func (srv *DHCPServer) Leases() (leases []*Lease) {
|
|
srv.leasesMu.RLock()
|
|
defer srv.leasesMu.RUnlock()
|
|
|
|
for _, iface := range srv.interfaces4 {
|
|
for _, lease := range iface.leases {
|
|
leases = append(leases, lease.Clone())
|
|
}
|
|
}
|
|
for _, iface := range srv.interfaces6 {
|
|
for _, lease := range iface.leases {
|
|
leases = append(leases, lease.Clone())
|
|
}
|
|
}
|
|
|
|
return leases
|
|
}
|
|
|
|
// HostByIP implements the [Interface] interface for *DHCPServer.
|
|
func (srv *DHCPServer) HostByIP(ip netip.Addr) (host string) {
|
|
srv.leasesMu.RLock()
|
|
defer srv.leasesMu.RUnlock()
|
|
|
|
if l, ok := srv.leases.leaseByAddr(ip); ok {
|
|
return l.Hostname
|
|
}
|
|
|
|
return ""
|
|
}
|
|
|
|
// MACByIP implements the [Interface] interface for *DHCPServer.
|
|
func (srv *DHCPServer) MACByIP(ip netip.Addr) (mac net.HardwareAddr) {
|
|
srv.leasesMu.RLock()
|
|
defer srv.leasesMu.RUnlock()
|
|
|
|
if l, ok := srv.leases.leaseByAddr(ip); ok {
|
|
return l.HWAddr
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// IPByHost implements the [Interface] interface for *DHCPServer.
|
|
func (srv *DHCPServer) IPByHost(host string) (ip netip.Addr) {
|
|
srv.leasesMu.RLock()
|
|
defer srv.leasesMu.RUnlock()
|
|
|
|
if l, ok := srv.leases.leaseByName(host); ok {
|
|
return l.IP
|
|
}
|
|
|
|
return netip.Addr{}
|
|
}
|
|
|
|
// Reset implements the [Interface] interface for *DHCPServer.
|
|
func (srv *DHCPServer) Reset(ctx context.Context) (err error) {
|
|
srv.leasesMu.Lock()
|
|
defer srv.leasesMu.Unlock()
|
|
|
|
for _, iface := range srv.interfaces4 {
|
|
iface.reset()
|
|
}
|
|
for _, iface := range srv.interfaces6 {
|
|
iface.reset()
|
|
}
|
|
srv.leases.clear()
|
|
|
|
srv.logger.DebugContext(ctx, "reset leases")
|
|
|
|
return nil
|
|
}
|
|
|
|
// AddLease implements the [Interface] interface for *DHCPServer.
|
|
func (srv *DHCPServer) AddLease(ctx context.Context, l *Lease) (err error) {
|
|
defer func() { err = errors.Annotate(err, "adding lease: %w") }()
|
|
|
|
addr := l.IP
|
|
iface, err := srv.ifaceForAddr(addr)
|
|
if err != nil {
|
|
// Don't wrap the error since there is already an annotation deferred.
|
|
return err
|
|
}
|
|
|
|
srv.leasesMu.Lock()
|
|
defer srv.leasesMu.Unlock()
|
|
|
|
err = srv.leases.add(l, iface)
|
|
if err != nil {
|
|
// Don't wrap the error since there is already an annotation deferred.
|
|
return err
|
|
}
|
|
|
|
iface.logger.DebugContext(
|
|
ctx, "added lease",
|
|
"hostname", l.Hostname,
|
|
"ip", l.IP,
|
|
"mac", l.HWAddr,
|
|
"static", l.IsStatic,
|
|
)
|
|
|
|
return nil
|
|
}
|
|
|
|
// UpdateStaticLease implements the [Interface] interface for *DHCPServer.
|
|
//
|
|
// TODO(e.burkov): Support moving leases between interfaces.
|
|
func (srv *DHCPServer) UpdateStaticLease(ctx context.Context, l *Lease) (err error) {
|
|
defer func() { err = errors.Annotate(err, "updating static lease: %w") }()
|
|
|
|
addr := l.IP
|
|
iface, err := srv.ifaceForAddr(addr)
|
|
if err != nil {
|
|
// Don't wrap the error since there is already an annotation deferred.
|
|
return err
|
|
}
|
|
|
|
srv.leasesMu.Lock()
|
|
defer srv.leasesMu.Unlock()
|
|
|
|
err = srv.leases.update(l, iface)
|
|
if err != nil {
|
|
// Don't wrap the error since there is already an annotation deferred.
|
|
return err
|
|
}
|
|
|
|
iface.logger.DebugContext(
|
|
ctx, "updated lease",
|
|
"hostname", l.Hostname,
|
|
"ip", l.IP,
|
|
"mac", l.HWAddr,
|
|
"static", l.IsStatic,
|
|
)
|
|
|
|
return nil
|
|
}
|
|
|
|
// RemoveLease implements the [Interface] interface for *DHCPServer.
|
|
func (srv *DHCPServer) RemoveLease(ctx context.Context, l *Lease) (err error) {
|
|
defer func() { err = errors.Annotate(err, "removing lease: %w") }()
|
|
|
|
addr := l.IP
|
|
iface, err := srv.ifaceForAddr(addr)
|
|
if err != nil {
|
|
// Don't wrap the error since there is already an annotation deferred.
|
|
return err
|
|
}
|
|
|
|
srv.leasesMu.Lock()
|
|
defer srv.leasesMu.Unlock()
|
|
|
|
err = srv.leases.remove(l, iface)
|
|
if err != nil {
|
|
// Don't wrap the error since there is already an annotation deferred.
|
|
return err
|
|
}
|
|
|
|
iface.logger.DebugContext(
|
|
ctx, "removed lease",
|
|
"hostname", l.Hostname,
|
|
"ip", l.IP,
|
|
"mac", l.HWAddr,
|
|
"static", l.IsStatic,
|
|
)
|
|
|
|
return nil
|
|
}
|
|
|
|
// ifaceForAddr returns the handled network interface for the given IP address,
|
|
// or an error if no such interface exists.
|
|
func (srv *DHCPServer) ifaceForAddr(addr netip.Addr) (iface *netInterface, err error) {
|
|
var ok bool
|
|
if addr.Is4() {
|
|
iface, ok = srv.interfaces4.find(addr)
|
|
} else {
|
|
iface, ok = srv.interfaces6.find(addr)
|
|
}
|
|
if !ok {
|
|
return nil, fmt.Errorf("no interface for ip %s", addr)
|
|
}
|
|
|
|
return iface, nil
|
|
}
|