netutils_linux.go 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. // Copyright 2019 Yunion
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. package netutils2
  15. import (
  16. "fmt"
  17. "net"
  18. "strings"
  19. "github.com/vishvananda/netlink"
  20. "yunion.io/x/log"
  21. "yunion.io/x/pkg/errors"
  22. "yunion.io/x/onecloud/pkg/util/iproute2"
  23. "yunion.io/x/onecloud/pkg/util/procutils"
  24. )
  25. type SNicAddress struct {
  26. Addr string
  27. MaskLen int
  28. }
  29. func (n *SNetInterface) GetAddresses() []SNicAddress {
  30. addrList := iproute2.NewAddress(n.name)
  31. addrs4 := n.getAddresses(addrList.List4)
  32. addrs6 := n.getAddresses(addrList.List6)
  33. if len(addrs6) > 0 {
  34. addrs4 = append(addrs4, addrs6...)
  35. }
  36. return addrs4
  37. }
  38. func (n *SNetInterface) getAddresses(listFunc func() ([]net.IPNet, error)) []SNicAddress {
  39. ipnets, err := listFunc()
  40. if err != nil {
  41. log.Errorf("list address %s: %v", n.name, err)
  42. return nil
  43. }
  44. r := make([]SNicAddress, len(ipnets))
  45. for i, ipnet := range ipnets {
  46. ip := ipnet.IP
  47. masklen, _ := ipnet.Mask.Size()
  48. r[i] = SNicAddress{
  49. Addr: ip.String(),
  50. MaskLen: masklen,
  51. }
  52. }
  53. return r
  54. }
  55. func (n *SNetInterface) GetRouteSpecs() []iproute2.RouteSpec {
  56. routeList := iproute2.NewRoute(n.name)
  57. rets := make([]iproute2.RouteSpec, 0)
  58. routes4 := getRouteSpecs(routeList.List4)
  59. for i := range routes4 {
  60. if routes4[i].Gw == nil {
  61. continue
  62. }
  63. if strings.HasPrefix(routes4[i].Dst.String(), "169.") {
  64. continue
  65. }
  66. rets = append(rets, routes4[i])
  67. }
  68. routes6 := getRouteSpecs(routeList.List6)
  69. for i := range routes6 {
  70. isDefaultRoute := routes6[i].Dst.String() == "::/0"
  71. if !isDefaultRoute && routes6[i].Gw == nil {
  72. continue
  73. }
  74. if strings.HasPrefix(routes6[i].Dst.String(), "fe80:") {
  75. continue
  76. }
  77. rets = append(rets, routes6[i])
  78. }
  79. return rets
  80. }
  81. func (n *SNetInterface) Shutdown() error {
  82. return n.setStatus("down")
  83. }
  84. func (n *SNetInterface) Bringup() error {
  85. return n.setStatus("up")
  86. }
  87. func (n *SNetInterface) Reset() error {
  88. err := n.Shutdown()
  89. if err != nil {
  90. return errors.Wrap(err, "shutdown")
  91. }
  92. return n.Bringup()
  93. }
  94. func (n *SNetInterface) setStatus(status string) error {
  95. cmd := procutils.NewCommand("ip", "link", "set", n.name, status)
  96. msg, err := cmd.Output()
  97. if err != nil {
  98. return errors.Wrap(err, strings.TrimSpace(string(msg)))
  99. }
  100. return nil
  101. }
  102. func (n *SNetInterface) FlushAddrs() error {
  103. cmd := procutils.NewCommand("ip", "addr", "flush", "dev", n.name)
  104. msg, err := cmd.Output()
  105. if err != nil {
  106. return errors.Wrap(err, strings.TrimSpace(string(msg)))
  107. }
  108. return nil
  109. }
  110. func getRouteSpecs(listFunc func() ([]iproute2.RouteSpec, error)) []iproute2.RouteSpec {
  111. routespecs, err := listFunc()
  112. if err != nil {
  113. return nil
  114. }
  115. return routespecs
  116. }
  117. func DefaultSrcIpDev() (srcIp net.IP, ifname string, err error) {
  118. routes, err := iproute2.RouteGetByDst("114.114.114.114")
  119. if err != nil {
  120. err = errors.Wrap(err, "get route")
  121. return
  122. }
  123. if len(routes) == 0 {
  124. err = fmt.Errorf("no route")
  125. return
  126. }
  127. var errs []error
  128. for i := range routes {
  129. route := &routes[i]
  130. if len(route.Src) == 0 {
  131. continue
  132. }
  133. ip4 := route.Src.To4()
  134. if len(ip4) != 4 || ip4.Equal(net.IPv4zero) {
  135. errs = append(errs, fmt.Errorf("bad src ipv4 address: %s", ip4))
  136. continue
  137. }
  138. link, err2 := netlink.LinkByIndex(route.LinkIndex)
  139. if err2 != nil {
  140. errs = append(errs, errors.Wrap(err2, "link by index"))
  141. continue
  142. }
  143. srcIp = ip4
  144. ifname = link.Attrs().Name
  145. return
  146. }
  147. err = errors.NewAggregate(errs)
  148. return
  149. }