coredhcp.go 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279
  1. package coredhcp
  2. import (
  3. "errors"
  4. "fmt"
  5. "net"
  6. "github.com/coredhcp/coredhcp/config"
  7. "github.com/coredhcp/coredhcp/handler"
  8. "github.com/coredhcp/coredhcp/logger"
  9. "github.com/coredhcp/coredhcp/plugins"
  10. "github.com/insomniacslk/dhcp/dhcpv4"
  11. "github.com/insomniacslk/dhcp/dhcpv4/server4"
  12. "github.com/insomniacslk/dhcp/dhcpv6"
  13. "github.com/insomniacslk/dhcp/dhcpv6/server6"
  14. )
  15. var log = logger.GetLogger("coredhcp")
  16. // Server is a CoreDHCP server structure that holds information about
  17. // DHCPv6 and DHCPv4 servers, and their respective handlers.
  18. type Server struct {
  19. Handlers6 []handler.Handler6
  20. Handlers4 []handler.Handler4
  21. Config *config.Config
  22. Server6 *server6.Server
  23. Server4 *server4.Server
  24. errors chan error
  25. }
  26. // LoadPlugins reads a Config object and loads the plugins as specified in the
  27. // `plugins` section, in order. For a plugin to be available, it must have been
  28. // previously registered with plugins.RegisterPlugin. This is normally done at
  29. // plugin import time.
  30. // This function returns the list of loaded v6 plugins, the list of loaded v4
  31. // plugins, and an error if any.
  32. func (s *Server) LoadPlugins(conf *config.Config) ([]*plugins.Plugin, []*plugins.Plugin, error) {
  33. log.Print("Loading plugins...")
  34. loadedPlugins6 := make([]*plugins.Plugin, 0)
  35. loadedPlugins4 := make([]*plugins.Plugin, 0)
  36. if conf.Server6 == nil && conf.Server4 == nil {
  37. return nil, nil, errors.New("no configuration found for either DHCPv6 or DHCPv4")
  38. }
  39. // now load the plugins. We need to call its setup function with
  40. // the arguments extracted above. The setup function is mapped in
  41. // plugins.RegisteredPlugins .
  42. // Load DHCPv6 plugins.
  43. if conf.Server6 != nil {
  44. for _, pluginConf := range conf.Server6.Plugins {
  45. if plugin, ok := plugins.RegisteredPlugins[pluginConf.Name]; ok {
  46. log.Printf("DHCPv6: loading plugin `%s`", pluginConf.Name)
  47. if plugin.Setup6 == nil {
  48. log.Warningf("DHCPv6: plugin `%s` has no setup function for DHCPv6", pluginConf.Name)
  49. continue
  50. }
  51. h6, err := plugin.Setup6(pluginConf.Args...)
  52. if err != nil {
  53. return nil, nil, err
  54. }
  55. loadedPlugins6 = append(loadedPlugins6, plugin)
  56. if h6 == nil {
  57. return nil, nil, config.ConfigErrorFromString("no DHCPv6 handler for plugin %s", pluginConf.Name)
  58. }
  59. s.Handlers6 = append(s.Handlers6, h6)
  60. } else {
  61. return nil, nil, config.ConfigErrorFromString("DHCPv6: unknown plugin `%s`", pluginConf.Name)
  62. }
  63. }
  64. }
  65. // Load DHCPv4 plugins. Yes, duplicated code, there's not really much that
  66. // can be deduplicated here.
  67. if conf.Server4 != nil {
  68. for _, pluginConf := range conf.Server4.Plugins {
  69. if plugin, ok := plugins.RegisteredPlugins[pluginConf.Name]; ok {
  70. log.Printf("DHCPv4: loading plugin `%s`", pluginConf.Name)
  71. if plugin.Setup4 == nil {
  72. log.Warningf("DHCPv4: plugin `%s` has no setup function for DHCPv4", pluginConf.Name)
  73. continue
  74. }
  75. h4, err := plugin.Setup4(pluginConf.Args...)
  76. if err != nil {
  77. return nil, nil, err
  78. }
  79. loadedPlugins4 = append(loadedPlugins4, plugin)
  80. if h4 == nil {
  81. return nil, nil, config.ConfigErrorFromString("no DHCPv4 handler for plugin %s", pluginConf.Name)
  82. }
  83. s.Handlers4 = append(s.Handlers4, h4)
  84. //s.Handlers4 = append(s.Handlers4, h4)
  85. } else {
  86. return nil, nil, config.ConfigErrorFromString("DHCPv4: unknown plugin `%s`", pluginConf.Name)
  87. }
  88. }
  89. }
  90. return loadedPlugins6, loadedPlugins4, nil
  91. }
  92. // MainHandler6 runs for every received DHCPv6 packet. It will run every
  93. // registered handler in sequence, and reply with the resulting response.
  94. // It will not reply if the resulting response is `nil`.
  95. func (s *Server) MainHandler6(conn net.PacketConn, peer net.Addr, req dhcpv6.DHCPv6) {
  96. var (
  97. resp dhcpv6.DHCPv6
  98. stop bool
  99. err error
  100. )
  101. // decapsulate the relay message
  102. msg, err := req.GetInnerMessage()
  103. if err != nil {
  104. log.Warningf("DHCPv6: cannot get inner message: %v", err)
  105. return
  106. }
  107. // Create a suitable basic response packet
  108. switch msg.Type() {
  109. case dhcpv6.MessageTypeSolicit:
  110. if msg.GetOneOption(dhcpv6.OptionRapidCommit) != nil {
  111. resp, err = dhcpv6.NewReplyFromMessage(msg)
  112. } else {
  113. resp, err = dhcpv6.NewAdvertiseFromSolicit(msg)
  114. }
  115. case dhcpv6.MessageTypeRequest, dhcpv6.MessageTypeConfirm, dhcpv6.MessageTypeRenew,
  116. dhcpv6.MessageTypeRebind, dhcpv6.MessageTypeRelease, dhcpv6.MessageTypeInformationRequest:
  117. resp, err = dhcpv6.NewReplyFromMessage(msg)
  118. default:
  119. err = fmt.Errorf("MainHandler6: message type %d not supported", msg.Type())
  120. }
  121. if err != nil {
  122. log.Printf("MainHandler6: NewReplyFromDHCPv6Message failed: %v", err)
  123. return
  124. }
  125. for _, handler := range s.Handlers6 {
  126. resp, stop = handler(req, resp)
  127. if stop {
  128. break
  129. }
  130. }
  131. if resp == nil {
  132. log.Print("MainHandler6: dropping request because response is nil")
  133. return
  134. }
  135. // if the request was relayed, re-encapsulate the response
  136. if req.IsRelay() {
  137. tmp, err := dhcpv6.NewRelayReplFromRelayForw(req.(*dhcpv6.RelayMessage), resp.(*dhcpv6.Message))
  138. if err != nil {
  139. log.Warningf("DHCPv6: cannot create relay-repl from relay-forw: %v", err)
  140. return
  141. }
  142. resp = tmp
  143. }
  144. if _, err := conn.WriteTo(resp.ToBytes(), peer); err != nil {
  145. log.Printf("MainHandler6: conn.Write to %v failed: %v", peer, err)
  146. }
  147. }
  148. // MainHandler4 is like MainHandler6, but for DHCPv4 packets.
  149. func (s *Server) MainHandler4(conn net.PacketConn, _peer net.Addr, req *dhcpv4.DHCPv4) {
  150. var (
  151. resp, tmp *dhcpv4.DHCPv4
  152. err error
  153. stop bool
  154. )
  155. if req.OpCode != dhcpv4.OpcodeBootRequest {
  156. log.Printf("MainHandler4: unsupported opcode %d. Only BootRequest (%d) is supported", req.OpCode, dhcpv4.OpcodeBootRequest)
  157. return
  158. }
  159. tmp, err = dhcpv4.NewReplyFromRequest(req)
  160. if err != nil {
  161. log.Printf("MainHandler4: failed to build reply: %v", err)
  162. return
  163. }
  164. switch mt := req.MessageType(); mt {
  165. case dhcpv4.MessageTypeDiscover:
  166. tmp.UpdateOption(dhcpv4.OptMessageType(dhcpv4.MessageTypeOffer))
  167. case dhcpv4.MessageTypeRequest:
  168. tmp.UpdateOption(dhcpv4.OptMessageType(dhcpv4.MessageTypeAck))
  169. default:
  170. log.Printf("plugins/server: Unhandled message type: %v", mt)
  171. return
  172. }
  173. resp = tmp
  174. for _, handler := range s.Handlers4 {
  175. resp, stop = handler(req, resp)
  176. if stop {
  177. break
  178. }
  179. }
  180. if resp != nil {
  181. var peer net.Addr
  182. if !req.GatewayIPAddr.IsUnspecified() {
  183. // TODO: make RFC8357 compliant
  184. peer = &net.UDPAddr{IP: req.GatewayIPAddr, Port: dhcpv4.ServerPort}
  185. } else if resp.MessageType() == dhcpv4.MessageTypeNak {
  186. peer = &net.UDPAddr{IP: net.IPv4bcast, Port: dhcpv4.ClientPort}
  187. } else if !req.ClientIPAddr.IsUnspecified() {
  188. peer = &net.UDPAddr{IP: req.ClientIPAddr, Port: dhcpv4.ClientPort}
  189. } else if req.IsBroadcast() {
  190. peer = &net.UDPAddr{IP: net.IPv4bcast, Port: dhcpv4.ClientPort}
  191. } else {
  192. // FIXME: we're supposed to unicast to a specific *L2* address, and an L3
  193. // address that's not yet assigned.
  194. // I don't know how to do that with this API...
  195. //peer = &net.UDPAddr{IP: resp.YourIPAddr, Port: dhcpv4.ClientPort}
  196. log.Warn("Cannot handle non-broadcast-capable unspecified peers in an RFC-compliant way. " +
  197. "Response will be broadcast")
  198. peer = &net.UDPAddr{IP: net.IPv4bcast, Port: dhcpv4.ClientPort}
  199. }
  200. if _, err := conn.WriteTo(resp.ToBytes(), peer); err != nil {
  201. log.Printf("MainHandler4: conn.Write to %v failed: %v", peer, err)
  202. }
  203. } else {
  204. log.Print("MainHandler4: dropping request because response is nil")
  205. }
  206. }
  207. // Start will start the server asynchronously. See `Wait` to wait until
  208. // the execution ends.
  209. func (s *Server) Start() error {
  210. _, _, err := s.LoadPlugins(s.Config)
  211. if err != nil {
  212. return err
  213. }
  214. // listen
  215. if s.Config.Server6 != nil {
  216. log.Printf("Starting DHCPv6 listener on %v", s.Config.Server6.Listener)
  217. s.Server6, err = server6.NewServer(s.Config.Server6.Interface, s.Config.Server6.Listener, s.MainHandler6)
  218. if err != nil {
  219. return err
  220. }
  221. go func() {
  222. s.errors <- s.Server6.Serve()
  223. }()
  224. }
  225. if s.Config.Server4 != nil {
  226. log.Printf("Starting DHCPv4 listener on %v", s.Config.Server4.Listener)
  227. s.Server4, err = server4.NewServer(s.Config.Server4.Interface, s.Config.Server4.Listener, s.MainHandler4)
  228. if err != nil {
  229. return err
  230. }
  231. go func() {
  232. s.errors <- s.Server4.Serve()
  233. }()
  234. }
  235. return nil
  236. }
  237. // Wait waits until the end of the execution of the server.
  238. func (s *Server) Wait() error {
  239. log.Print("Waiting")
  240. err := <-s.errors
  241. if s.Server6 != nil {
  242. s.Server6.Close()
  243. }
  244. if s.Server4 != nil {
  245. s.Server4.Close()
  246. }
  247. return err
  248. }
  249. // NewServer creates a Server instance with the provided configuration.
  250. func NewServer(config *config.Config) *Server {
  251. return &Server{Config: config, errors: make(chan error, 1)}
  252. }