2023-06-22 14:18:43 +03:00
|
|
|
package dhcpsvc
|
|
|
|
|
|
|
|
import (
|
2023-10-02 13:21:16 +03:00
|
|
|
"fmt"
|
2024-07-03 15:29:54 +03:00
|
|
|
"log/slog"
|
2024-07-09 20:04:24 +03:00
|
|
|
"os"
|
2023-06-22 14:18:43 +03:00
|
|
|
"time"
|
|
|
|
|
2024-07-03 15:29:54 +03:00
|
|
|
"github.com/AdguardTeam/golibs/errors"
|
|
|
|
"github.com/AdguardTeam/golibs/mapsutil"
|
2023-10-02 13:21:16 +03:00
|
|
|
"github.com/AdguardTeam/golibs/netutil"
|
2023-06-22 14:18:43 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
// Config is the configuration for the DHCP service.
|
|
|
|
type Config struct {
|
|
|
|
// Interfaces stores configurations of DHCP server specific for the network
|
|
|
|
// interface identified by its name.
|
|
|
|
Interfaces map[string]*InterfaceConfig
|
|
|
|
|
2024-07-03 15:29:54 +03:00
|
|
|
// Logger will be used to log the DHCP events.
|
|
|
|
Logger *slog.Logger
|
|
|
|
|
2023-06-22 14:18:43 +03:00
|
|
|
// LocalDomainName is the top-level domain name to use for resolving DHCP
|
|
|
|
// clients' hostnames.
|
|
|
|
LocalDomainName string
|
|
|
|
|
2024-07-09 20:04:24 +03:00
|
|
|
// DBFilePath is the path to the database file containing the DHCP leases.
|
|
|
|
DBFilePath string
|
2024-01-31 14:50:27 +03:00
|
|
|
|
2023-06-22 14:18:43 +03:00
|
|
|
// ICMPTimeout is the timeout for checking another DHCP server's presence.
|
|
|
|
ICMPTimeout time.Duration
|
|
|
|
|
|
|
|
// Enabled is the state of the service, whether it is enabled or not.
|
|
|
|
Enabled bool
|
|
|
|
}
|
|
|
|
|
|
|
|
// InterfaceConfig is the configuration of a single DHCP interface.
|
|
|
|
type InterfaceConfig struct {
|
|
|
|
// IPv4 is the configuration of DHCP protocol for IPv4.
|
|
|
|
IPv4 *IPv4Config
|
|
|
|
|
|
|
|
// IPv6 is the configuration of DHCP protocol for IPv6.
|
|
|
|
IPv6 *IPv6Config
|
|
|
|
}
|
|
|
|
|
2023-10-02 13:21:16 +03:00
|
|
|
// Validate returns an error in conf if any.
|
2024-07-03 15:29:54 +03:00
|
|
|
//
|
|
|
|
// TODO(e.burkov): Unexport and rewrite the test.
|
2023-10-02 13:21:16 +03:00
|
|
|
func (conf *Config) Validate() (err error) {
|
|
|
|
switch {
|
|
|
|
case conf == nil:
|
|
|
|
return errNilConfig
|
|
|
|
case !conf.Enabled:
|
|
|
|
return nil
|
2024-07-03 15:29:54 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
var errs []error
|
|
|
|
if conf.ICMPTimeout < 0 {
|
|
|
|
err = newMustErr("icmp timeout", "be non-negative", conf.ICMPTimeout)
|
|
|
|
errs = append(errs, err)
|
2023-10-02 13:21:16 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
err = netutil.ValidateDomainName(conf.LocalDomainName)
|
|
|
|
if err != nil {
|
|
|
|
// Don't wrap the error since it's informative enough as is.
|
2024-07-03 15:29:54 +03:00
|
|
|
errs = append(errs, err)
|
2023-10-02 13:21:16 +03:00
|
|
|
}
|
|
|
|
|
2024-07-09 20:04:24 +03:00
|
|
|
// This is a best-effort check for the file accessibility. The file will be
|
|
|
|
// checked again when it is opened later.
|
|
|
|
if _, err = os.Stat(conf.DBFilePath); err != nil && !errors.Is(err, os.ErrNotExist) {
|
|
|
|
errs = append(errs, fmt.Errorf("db file path %q: %w", conf.DBFilePath, err))
|
|
|
|
}
|
|
|
|
|
2023-10-02 13:21:16 +03:00
|
|
|
if len(conf.Interfaces) == 0 {
|
2024-07-03 15:29:54 +03:00
|
|
|
errs = append(errs, errNoInterfaces)
|
2023-10-02 13:21:16 +03:00
|
|
|
|
2024-07-03 15:29:54 +03:00
|
|
|
return errors.Join(errs...)
|
|
|
|
}
|
2023-10-02 13:21:16 +03:00
|
|
|
|
2024-07-03 15:29:54 +03:00
|
|
|
mapsutil.SortedRange(conf.Interfaces, func(iface string, ic *InterfaceConfig) (ok bool) {
|
|
|
|
err = ic.validate()
|
|
|
|
if err != nil {
|
|
|
|
errs = append(errs, fmt.Errorf("interface %q: %w", iface, err))
|
2023-10-02 13:21:16 +03:00
|
|
|
}
|
|
|
|
|
2024-07-03 15:29:54 +03:00
|
|
|
return true
|
|
|
|
})
|
|
|
|
|
|
|
|
return errors.Join(errs...)
|
2023-06-22 14:18:43 +03:00
|
|
|
}
|
|
|
|
|
2023-10-02 13:21:16 +03:00
|
|
|
// validate returns an error in ic, if any.
|
|
|
|
func (ic *InterfaceConfig) validate() (err error) {
|
|
|
|
if ic == nil {
|
|
|
|
return errNilConfig
|
|
|
|
}
|
2023-06-22 14:18:43 +03:00
|
|
|
|
2023-10-02 13:21:16 +03:00
|
|
|
if err = ic.IPv4.validate(); err != nil {
|
|
|
|
return fmt.Errorf("ipv4: %w", err)
|
|
|
|
}
|
2023-06-22 14:18:43 +03:00
|
|
|
|
2023-10-02 13:21:16 +03:00
|
|
|
if err = ic.IPv6.validate(); err != nil {
|
|
|
|
return fmt.Errorf("ipv6: %w", err)
|
|
|
|
}
|
2023-06-22 14:18:43 +03:00
|
|
|
|
2023-10-02 13:21:16 +03:00
|
|
|
return nil
|
2023-06-22 14:18:43 +03:00
|
|
|
}
|