serve.go 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160
  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 server
  5. import (
  6. "fmt"
  7. "io"
  8. "net"
  9. "golang.org/x/net/ipv4"
  10. "golang.org/x/net/ipv6"
  11. "github.com/coredhcp/coredhcp/config"
  12. "github.com/coredhcp/coredhcp/handler"
  13. "github.com/coredhcp/coredhcp/logger"
  14. "github.com/coredhcp/coredhcp/plugins"
  15. "github.com/insomniacslk/dhcp/dhcpv4/server4"
  16. "github.com/insomniacslk/dhcp/dhcpv6/server6"
  17. )
  18. var log = logger.GetLogger("server")
  19. type listener6 struct {
  20. *ipv6.PacketConn
  21. handlers []handler.Handler6
  22. }
  23. type listener4 struct {
  24. *ipv4.PacketConn
  25. handlers []handler.Handler4
  26. }
  27. type listener interface {
  28. io.Closer
  29. }
  30. // Servers contains state for a running server (with possibly multiple interfaces/listeners)
  31. type Servers struct {
  32. listeners []listener
  33. errors chan error
  34. }
  35. func listen4(a *net.UDPAddr) (*listener4, error) {
  36. var err error
  37. l4 := listener4{}
  38. udpConn, err := server4.NewIPv4UDPConn(a.Zone, a)
  39. if err != nil {
  40. return nil, err
  41. }
  42. l4.PacketConn = ipv4.NewPacketConn(udpConn)
  43. var ifi *net.Interface
  44. if a.Zone != "" {
  45. ifi, err = net.InterfaceByName(a.Zone)
  46. if err != nil {
  47. return nil, fmt.Errorf("DHCPv4: Listen could not find interface %s: %v", a.Zone, err)
  48. }
  49. }
  50. if a.IP.IsMulticast() {
  51. err = l4.JoinGroup(ifi, a)
  52. if err != nil {
  53. return nil, err
  54. }
  55. }
  56. return &l4, nil
  57. }
  58. func listen6(a *net.UDPAddr) (*listener6, error) {
  59. l6 := listener6{}
  60. udpconn, err := server6.NewIPv6UDPConn(a.Zone, a)
  61. if err != nil {
  62. return nil, err
  63. }
  64. l6.PacketConn = ipv6.NewPacketConn(udpconn)
  65. var ifi *net.Interface
  66. if a.Zone != "" {
  67. ifi, err = net.InterfaceByName(a.Zone)
  68. if err != nil {
  69. return nil, fmt.Errorf("DHCPv4: Listen could not find interface %s: %v", a.Zone, err)
  70. }
  71. }
  72. if a.IP.IsMulticast() {
  73. err = l6.JoinGroup(ifi, a)
  74. if err != nil {
  75. return nil, err
  76. }
  77. }
  78. return &l6, nil
  79. }
  80. // Start will start the server asynchronously. See `Wait` to wait until
  81. // the execution ends.
  82. func Start(config *config.Config) (*Servers, error) {
  83. handlers4, handlers6, err := plugins.LoadPlugins(config)
  84. if err != nil {
  85. return nil, err
  86. }
  87. srv := Servers{
  88. errors: make(chan error),
  89. }
  90. // listen
  91. if config.Server6 != nil {
  92. log.Println("Starting DHCPv6 server")
  93. for _, addr := range config.Server6.Addresses {
  94. var l6 *listener6
  95. l6, err = listen6(addr)
  96. if err != nil {
  97. goto cleanup
  98. }
  99. l6.handlers = handlers6
  100. srv.listeners = append(srv.listeners, l6)
  101. go func() {
  102. srv.errors <- l6.Serve()
  103. }()
  104. }
  105. }
  106. if config.Server4 != nil {
  107. log.Println("Starting DHCPv4 server")
  108. for _, addr := range config.Server4.Addresses {
  109. var l4 *listener4
  110. l4, err = listen4(addr)
  111. if err != nil {
  112. goto cleanup
  113. }
  114. l4.handlers = handlers4
  115. srv.listeners = append(srv.listeners, l4)
  116. go func() {
  117. srv.errors <- l4.Serve()
  118. }()
  119. }
  120. }
  121. return &srv, nil
  122. cleanup:
  123. srv.Close()
  124. return nil, err
  125. }
  126. // Wait waits until the end of the execution of the server.
  127. func (s *Servers) Wait() error {
  128. log.Debug("Waiting")
  129. err := <-s.errors
  130. s.Close()
  131. return err
  132. }
  133. // Close closes all listening connections
  134. func (s *Servers) Close() {
  135. for _, srv := range s.listeners {
  136. if srv != nil {
  137. srv.Close()
  138. }
  139. }
  140. }