coredhcp.go 8.7 KB

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