interfaces.go 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197
  1. package interfaces
  2. import (
  3. "fmt"
  4. "net"
  5. "os"
  6. "os/exec"
  7. "path/filepath"
  8. "sort"
  9. "strings"
  10. "networktool/internal/config"
  11. "networktool/internal/model"
  12. )
  13. type Service struct {
  14. cfg config.Config
  15. }
  16. func New(cfg config.Config) *Service {
  17. return &Service{cfg: cfg}
  18. }
  19. func (s *Service) List() (model.InterfacesResponse, error) {
  20. items, err := s.listPhysicalInterfaces()
  21. if err != nil {
  22. return model.InterfacesResponse{}, err
  23. }
  24. management := s.detectManagement(items)
  25. for i := range items {
  26. items[i].IsManagement = items[i].SystemName == management
  27. items[i].Name = items[i].SystemName
  28. if items[i].IsManagement {
  29. items[i].Role = "control"
  30. } else {
  31. items[i].Role = "business"
  32. }
  33. }
  34. suggested, requiresSelection := suggestTarget(items, management)
  35. for i := range items {
  36. items[i].IsSuggestedTarget = items[i].SystemName == suggested
  37. }
  38. return model.InterfacesResponse{
  39. ManagementInterface: management,
  40. SuggestedTargetInterface: suggested,
  41. RequiresTargetSelection: requiresSelection,
  42. Interfaces: items,
  43. }, nil
  44. }
  45. func (s *Service) EnsureMaintenanceAddress() error {
  46. items, err := s.listPhysicalInterfaces()
  47. if err != nil {
  48. return err
  49. }
  50. if len(items) == 0 {
  51. return fmt.Errorf("no physical ethernet interfaces found")
  52. }
  53. management := s.detectManagement(items)
  54. if management == "" {
  55. return fmt.Errorf("failed to detect management interface")
  56. }
  57. for _, item := range items {
  58. if item.SystemName != management {
  59. continue
  60. }
  61. for _, addr := range item.IPv4 {
  62. if addr.Address == s.cfg.MaintenanceIP {
  63. return nil
  64. }
  65. }
  66. break
  67. }
  68. cmd := exec.Command("ip", "addr", "add", s.cfg.MaintenanceCIDR, "dev", management)
  69. output, err := cmd.CombinedOutput()
  70. if err != nil {
  71. if strings.Contains(string(output), "File exists") {
  72. return nil
  73. }
  74. return fmt.Errorf("ip addr add failed on %s: %s", management, strings.TrimSpace(string(output)))
  75. }
  76. return nil
  77. }
  78. func (s *Service) listPhysicalInterfaces() ([]model.NetworkInterface, error) {
  79. ifaces, err := net.Interfaces()
  80. if err != nil {
  81. return nil, err
  82. }
  83. items := make([]model.NetworkInterface, 0)
  84. for _, iface := range ifaces {
  85. if !isPhysicalEthernet(iface.Name) {
  86. continue
  87. }
  88. item := model.NetworkInterface{
  89. SystemName: iface.Name,
  90. MAC: iface.HardwareAddr.String(),
  91. LinkUp: readLinkUp(iface.Name),
  92. Gateway: "",
  93. DNS: []string{},
  94. }
  95. item.IPv4 = readIPv4(iface)
  96. items = append(items, item)
  97. }
  98. sort.Slice(items, func(i, j int) bool { return items[i].SystemName < items[j].SystemName })
  99. return items, nil
  100. }
  101. func (s *Service) detectManagement(items []model.NetworkInterface) string {
  102. for _, item := range items {
  103. for _, addr := range item.IPv4 {
  104. if addr.Address == s.cfg.MaintenanceIP {
  105. return item.SystemName
  106. }
  107. }
  108. }
  109. if len(items) == 2 {
  110. return items[1].SystemName
  111. }
  112. if len(items) > 0 {
  113. return items[len(items)-1].SystemName
  114. }
  115. return ""
  116. }
  117. func suggestTarget(items []model.NetworkInterface, management string) (string, bool) {
  118. candidates := make([]string, 0)
  119. for _, item := range items {
  120. if item.SystemName != management {
  121. candidates = append(candidates, item.SystemName)
  122. }
  123. }
  124. if len(candidates) == 1 {
  125. return candidates[0], false
  126. }
  127. return "", len(candidates) > 1
  128. }
  129. func readIPv4(iface net.Interface) []model.IPv4Address {
  130. addrs, err := iface.Addrs()
  131. if err != nil {
  132. return nil
  133. }
  134. result := make([]model.IPv4Address, 0)
  135. for _, addr := range addrs {
  136. ipNet, ok := addr.(*net.IPNet)
  137. if !ok || ipNet.IP.To4() == nil {
  138. continue
  139. }
  140. prefix, _ := ipNet.Mask.Size()
  141. source := "unknown"
  142. if ipNet.IP.String() == "169.254.100.2" {
  143. source = "static"
  144. }
  145. result = append(result, model.IPv4Address{Address: ipNet.IP.String(), Prefix: prefix, Source: source})
  146. }
  147. return result
  148. }
  149. func isPhysicalEthernet(name string) bool {
  150. if name == "lo" {
  151. return false
  152. }
  153. for _, prefix := range []string{"docker", "br-", "veth", "virbr", "tun", "tap", "wg", "zt", "vmnet"} {
  154. if strings.HasPrefix(name, prefix) {
  155. return false
  156. }
  157. }
  158. devicePath := filepath.Join("/sys/class/net", name, "device")
  159. if _, err := os.Stat(devicePath); err != nil {
  160. return false
  161. }
  162. return true
  163. }
  164. func readLinkUp(name string) bool {
  165. carrierPath := filepath.Join("/sys/class/net", name, "carrier")
  166. data, err := os.ReadFile(carrierPath)
  167. if err == nil {
  168. return strings.TrimSpace(string(data)) == "1"
  169. }
  170. statePath := filepath.Join("/sys/class/net", name, "operstate")
  171. data, err = os.ReadFile(statePath)
  172. if err != nil {
  173. return false
  174. }
  175. return strings.TrimSpace(string(data)) == "up"
  176. }