Files
cloudpods/pkg/util/netutils2/netutils_linux.go

159 lines
3.6 KiB
Go

// Copyright 2019 Yunion
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package netutils2
import (
"fmt"
"net"
"strings"
"github.com/vishvananda/netlink"
"yunion.io/x/log"
"yunion.io/x/pkg/errors"
"yunion.io/x/onecloud/pkg/util/iproute2"
"yunion.io/x/onecloud/pkg/util/procutils"
)
type SNicAddress struct {
Addr string
MaskLen int
}
func (n *SNetInterface) GetAddresses() []SNicAddress {
addrList := iproute2.NewAddress(n.name)
addrs4 := n.getAddresses(addrList.List4)
addrs6 := n.getAddresses(addrList.List6)
if len(addrs6) > 0 {
addrs4 = append(addrs4, addrs6...)
}
return addrs4
}
func (n *SNetInterface) getAddresses(listFunc func() ([]net.IPNet, error)) []SNicAddress {
ipnets, err := listFunc()
if err != nil {
log.Errorf("list address %s: %v", n.name, err)
return nil
}
r := make([]SNicAddress, len(ipnets))
for i, ipnet := range ipnets {
ip := ipnet.IP
masklen, _ := ipnet.Mask.Size()
r[i] = SNicAddress{
Addr: ip.String(),
MaskLen: masklen,
}
}
return r
}
func (n *SNetInterface) GetRouteSpecs() []iproute2.RouteSpec {
routeList := iproute2.NewRoute(n.name)
rets := make([]iproute2.RouteSpec, 0)
routes4 := getRouteSpecs(routeList.List4)
for i := range routes4 {
if routes4[i].Gw == nil {
continue
}
if strings.HasPrefix(routes4[i].Dst.String(), "169.") {
continue
}
rets = append(rets, routes4[i])
}
routes6 := getRouteSpecs(routeList.List6)
for i := range routes6 {
isDefaultRoute := routes6[i].Dst.String() == "::/0"
if !isDefaultRoute && routes6[i].Gw == nil {
continue
}
if strings.HasPrefix(routes6[i].Dst.String(), "fe80:") {
continue
}
rets = append(rets, routes6[i])
}
return rets
}
func (n *SNetInterface) Shutdown() error {
return n.setStatus("down")
}
func (n *SNetInterface) Bringup() error {
return n.setStatus("up")
}
func (n *SNetInterface) Reset() error {
err := n.Shutdown()
if err != nil {
return errors.Wrap(err, "shutdown")
}
return n.Bringup()
}
func (n *SNetInterface) setStatus(status string) error {
cmd := procutils.NewCommand("ip", "link", "set", n.name, status)
msg, err := cmd.Output()
if err != nil {
return errors.Wrap(err, strings.TrimSpace(string(msg)))
}
return nil
}
func getRouteSpecs(listFunc func() ([]iproute2.RouteSpec, error)) []iproute2.RouteSpec {
routespecs, err := listFunc()
if err != nil {
return nil
}
return routespecs
}
func DefaultSrcIpDev() (srcIp net.IP, ifname string, err error) {
routes, err := iproute2.RouteGetByDst("114.114.114.114")
if err != nil {
err = errors.Wrap(err, "get route")
return
}
if len(routes) == 0 {
err = fmt.Errorf("no route")
return
}
var errs []error
for i := range routes {
route := &routes[i]
if len(route.Src) == 0 {
continue
}
ip4 := route.Src.To4()
if len(ip4) != 4 || ip4.Equal(net.IPv4zero) {
errs = append(errs, fmt.Errorf("bad src ipv4 address: %s", ip4))
continue
}
link, err2 := netlink.LinkByIndex(route.LinkIndex)
if err2 != nil {
errs = append(errs, errors.Wrap(err2, "link by index"))
continue
}
srcIp = ip4
ifname = link.Attrs().Name
return
}
err = errors.NewAggregate(errs)
return
}