| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247 |
- package coredhcp
- import (
- "errors"
- "fmt"
- "net"
- "github.com/coredhcp/coredhcp/config"
- "github.com/coredhcp/coredhcp/handler"
- "github.com/coredhcp/coredhcp/logger"
- "github.com/coredhcp/coredhcp/plugins"
- "github.com/insomniacslk/dhcp/dhcpv4"
- "github.com/insomniacslk/dhcp/dhcpv4/server4"
- "github.com/insomniacslk/dhcp/dhcpv6"
- "github.com/insomniacslk/dhcp/dhcpv6/server6"
- )
- var log = logger.GetLogger()
- // Server is a CoreDHCP server structure that holds information about
- // DHCPv6 and DHCPv4 servers, and their respective handlers.
- type Server struct {
- Handlers6 []handler.Handler6
- Handlers4 []handler.Handler4
- Config *config.Config
- Server6 *server6.Server
- Server4 *server4.Server
- errors chan error
- }
- // LoadPlugins reads a Config object and loads the plugins as specified in the
- // `plugins` section, in order. For a plugin to be available, it must have been
- // previously registered with plugins.RegisterPlugin. This is normally done at
- // plugin import time.
- // This function returns the list of loaded v6 plugins, the list of loaded v4
- // plugins, and an error if any.
- func (s *Server) LoadPlugins(conf *config.Config) ([]*plugins.Plugin, []*plugins.Plugin, error) {
- log.Print("Loading plugins...")
- loadedPlugins6 := make([]*plugins.Plugin, 0)
- loadedPlugins4 := make([]*plugins.Plugin, 0)
- if conf.Server6 == nil && conf.Server4 == nil {
- return nil, nil, errors.New("no configuration found for either DHCPv6 or DHCPv4")
- }
- // now load the plugins. We need to call its setup function with
- // the arguments extracted above. The setup function is mapped in
- // plugins.RegisteredPlugins .
- // Load DHCPv6 plugins.
- if conf.Server6 != nil {
- for _, pluginConf := range conf.Server6.Plugins {
- if plugin, ok := plugins.RegisteredPlugins[pluginConf.Name]; ok {
- log.Printf("DHCPv6: loading plugin `%s`", pluginConf.Name)
- if plugin.Setup6 == nil {
- log.Warningf("DHCPv6: plugin `%s` has no setup function for DHCPv6", pluginConf.Name)
- continue
- }
- h6, err := plugin.Setup6(pluginConf.Args...)
- if err != nil {
- return nil, nil, err
- }
- loadedPlugins6 = append(loadedPlugins6, plugin)
- if h6 == nil {
- return nil, nil, config.ConfigErrorFromString("no DHCPv6 handler for plugin %s", pluginConf.Name)
- }
- s.Handlers6 = append(s.Handlers6, h6)
- } else {
- return nil, nil, config.ConfigErrorFromString("DHCPv6: unknown plugin `%s`", pluginConf.Name)
- }
- }
- }
- // Load DHCPv4 plugins. Yes, duplicated code, there's not really much that
- // can be deduplicated here.
- if conf.Server4 != nil {
- for _, pluginConf := range conf.Server4.Plugins {
- if plugin, ok := plugins.RegisteredPlugins[pluginConf.Name]; ok {
- log.Printf("DHCPv4: loading plugin `%s`", pluginConf.Name)
- if plugin.Setup4 == nil {
- log.Warningf("DHCPv4: plugin `%s` has no setup function for DHCPv4", pluginConf.Name)
- continue
- }
- h4, err := plugin.Setup4(pluginConf.Args...)
- if err != nil {
- return nil, nil, err
- }
- loadedPlugins4 = append(loadedPlugins4, plugin)
- if h4 == nil {
- return nil, nil, config.ConfigErrorFromString("no DHCPv4 handler for plugin %s", pluginConf.Name)
- }
- s.Handlers4 = append(s.Handlers4, h4)
- //s.Handlers4 = append(s.Handlers4, h4)
- } else {
- return nil, nil, config.ConfigErrorFromString("DHCPv4: unknown plugin `%s`", pluginConf.Name)
- }
- }
- }
- return loadedPlugins6, loadedPlugins4, nil
- }
- // MainHandler6 runs for every received DHCPv6 packet. It will run every
- // registered handler in sequence, and reply with the resulting response.
- // It will not reply if the resulting response is `nil`.
- func (s *Server) MainHandler6(conn net.PacketConn, peer net.Addr, req dhcpv6.DHCPv6) {
- var (
- resp, tmp dhcpv6.DHCPv6
- stop bool
- err error
- )
- // Create a suitable basic response packet
- switch req.Type() {
- case dhcpv6.MessageTypeSolicit:
- tmp, err = dhcpv6.NewAdvertiseFromSolicit(req.(*dhcpv6.Message))
- case dhcpv6.MessageTypeRequest, dhcpv6.MessageTypeConfirm, dhcpv6.MessageTypeRenew,
- dhcpv6.MessageTypeRebind, dhcpv6.MessageTypeRelease, dhcpv6.MessageTypeInformationRequest:
- tmp, err = dhcpv6.NewReplyFromMessage(req.(*dhcpv6.Message))
- default:
- err = fmt.Errorf("MainHandler6: message type %d not supported", req.Type())
- }
- if err != nil {
- log.Printf("MainHandler6: NewReplyFromDHCPv6Message failed: %v", err)
- return
- }
- resp = tmp
- for _, handler := range s.Handlers6 {
- resp, stop = handler(req, resp)
- if stop {
- break
- }
- }
- if resp != nil {
- if _, err := conn.WriteTo(resp.ToBytes(), peer); err != nil {
- log.Printf("MainHandler6: conn.Write to %v failed: %v", peer, err)
- }
- } else {
- log.Print("MainHandler6: dropping request because response is nil")
- }
- }
- // MainHandler4 is like MainHandler6, but for DHCPv4 packets.
- func (s *Server) MainHandler4(conn net.PacketConn, peer net.Addr, req *dhcpv4.DHCPv4) {
- var (
- resp, tmp *dhcpv4.DHCPv4
- err error
- stop bool
- )
- if req.OpCode != dhcpv4.OpcodeBootRequest {
- log.Printf("MainHandler4: unsupported opcode %d. Only BootRequest (%d) is supported", req.OpCode, dhcpv4.OpcodeBootRequest)
- return
- }
- tmp, err = dhcpv4.NewReplyFromRequest(req)
- if err != nil {
- log.Printf("MainHandler4: failed to build reply: %v", err)
- return
- }
- resp = tmp
- for _, handler := range s.Handlers4 {
- resp, stop = handler(req, resp)
- if stop {
- break
- }
- }
- if resp != nil {
- if !req.GatewayIPAddr.IsUnspecified() {
- // TODO: make RFC8357 compliant
- peer = &net.UDPAddr{IP: req.GatewayIPAddr, Port: dhcpv4.ServerPort}
- }
- if _, err := conn.WriteTo(resp.ToBytes(), peer); err != nil {
- log.Printf("MainHandler4: conn.Write to %v failed: %v", peer, err)
- }
- } else {
- log.Print("MainHandler4: dropping request because response is nil")
- }
- }
- // Start will start the server asynchronously. See `Wait` to wait until
- // the execution ends.
- func (s *Server) Start() error {
- _, _, err := s.LoadPlugins(s.Config)
- if err != nil {
- return err
- }
- // listen
- if s.Config.Server6 != nil {
- log.Printf("Starting DHCPv6 listener on %v", s.Config.Server6.Listener)
- listener := s.Config.Server6.Listener
- ifname := s.Config.Server6.Interface
- opts := make([]server6.ServerOpt, 0)
- if listener.IP.IsMulticast() {
- iface, err := net.InterfaceByName(ifname)
- if err != nil {
- return err
- }
- conn, err := net.ListenMulticastUDP("udp6", iface, listener)
- if err != nil {
- return err
- }
- opts = append(opts, server6.WithConn(conn))
- }
- s.Server6, err = server6.NewServer(s.Config.Server6.Listener, s.MainHandler6, opts...)
- if err != nil {
- return err
- }
- go func() {
- s.errors <- s.Server6.Serve()
- }()
- }
- if s.Config.Server4 != nil {
- log.Printf("Starting DHCPv4 listener on %v", s.Config.Server4.Listener)
- s.Server4, err = server4.NewServer(s.Config.Server4.Listener, s.MainHandler4)
- if err != nil {
- return err
- }
- go func() {
- s.errors <- s.Server4.Serve()
- }()
- }
- return nil
- }
- // Wait waits until the end of the execution of the server.
- func (s *Server) Wait() error {
- log.Print("Waiting")
- err := <-s.errors
- if s.Server6 != nil {
- s.Server6.Close()
- }
- if s.Server4 != nil {
- s.Server4.Close()
- }
- return err
- }
- // NewServer creates a Server instance with the provided configuration.
- func NewServer(config *config.Config) *Server {
- return &Server{Config: config, errors: make(chan error, 1)}
- }
|