mirror of
				https://github.com/k3s-io/kubernetes.git
				synced 2025-11-03 23:40:03 +00:00 
			
		
		
		
	godep restore pushd $GOPATH/src/github.com/appc/spec git co master popd go get go4.org/errorutil rm -rf Godeps godep save ./... git add vendor git add -f $(git ls-files --other vendor/) git co -- Godeps/LICENSES Godeps/.license_file_state Godeps/OWNERS
		
			
				
	
	
		
			260 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			260 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
// +build linux
 | 
						|
 | 
						|
package libcontainer
 | 
						|
 | 
						|
import (
 | 
						|
	"fmt"
 | 
						|
	"io/ioutil"
 | 
						|
	"net"
 | 
						|
	"path/filepath"
 | 
						|
	"strconv"
 | 
						|
	"strings"
 | 
						|
 | 
						|
	"github.com/opencontainers/runc/libcontainer/configs"
 | 
						|
	"github.com/opencontainers/runc/libcontainer/utils"
 | 
						|
	"github.com/vishvananda/netlink"
 | 
						|
)
 | 
						|
 | 
						|
var strategies = map[string]networkStrategy{
 | 
						|
	"veth":     &veth{},
 | 
						|
	"loopback": &loopback{},
 | 
						|
}
 | 
						|
 | 
						|
// networkStrategy represents a specific network configuration for
 | 
						|
// a container's networking stack
 | 
						|
type networkStrategy interface {
 | 
						|
	create(*network, int) error
 | 
						|
	initialize(*network) error
 | 
						|
	detach(*configs.Network) error
 | 
						|
	attach(*configs.Network) error
 | 
						|
}
 | 
						|
 | 
						|
// getStrategy returns the specific network strategy for the
 | 
						|
// provided type.
 | 
						|
func getStrategy(tpe string) (networkStrategy, error) {
 | 
						|
	s, exists := strategies[tpe]
 | 
						|
	if !exists {
 | 
						|
		return nil, fmt.Errorf("unknown strategy type %q", tpe)
 | 
						|
	}
 | 
						|
	return s, nil
 | 
						|
}
 | 
						|
 | 
						|
// Returns the network statistics for the network interfaces represented by the NetworkRuntimeInfo.
 | 
						|
func getNetworkInterfaceStats(interfaceName string) (*NetworkInterface, error) {
 | 
						|
	out := &NetworkInterface{Name: interfaceName}
 | 
						|
	// This can happen if the network runtime information is missing - possible if the
 | 
						|
	// container was created by an old version of libcontainer.
 | 
						|
	if interfaceName == "" {
 | 
						|
		return out, nil
 | 
						|
	}
 | 
						|
	type netStatsPair struct {
 | 
						|
		// Where to write the output.
 | 
						|
		Out *uint64
 | 
						|
		// The network stats file to read.
 | 
						|
		File string
 | 
						|
	}
 | 
						|
	// Ingress for host veth is from the container. Hence tx_bytes stat on the host veth is actually number of bytes received by the container.
 | 
						|
	netStats := []netStatsPair{
 | 
						|
		{Out: &out.RxBytes, File: "tx_bytes"},
 | 
						|
		{Out: &out.RxPackets, File: "tx_packets"},
 | 
						|
		{Out: &out.RxErrors, File: "tx_errors"},
 | 
						|
		{Out: &out.RxDropped, File: "tx_dropped"},
 | 
						|
 | 
						|
		{Out: &out.TxBytes, File: "rx_bytes"},
 | 
						|
		{Out: &out.TxPackets, File: "rx_packets"},
 | 
						|
		{Out: &out.TxErrors, File: "rx_errors"},
 | 
						|
		{Out: &out.TxDropped, File: "rx_dropped"},
 | 
						|
	}
 | 
						|
	for _, netStat := range netStats {
 | 
						|
		data, err := readSysfsNetworkStats(interfaceName, netStat.File)
 | 
						|
		if err != nil {
 | 
						|
			return nil, err
 | 
						|
		}
 | 
						|
		*(netStat.Out) = data
 | 
						|
	}
 | 
						|
	return out, nil
 | 
						|
}
 | 
						|
 | 
						|
// Reads the specified statistics available under /sys/class/net/<EthInterface>/statistics
 | 
						|
func readSysfsNetworkStats(ethInterface, statsFile string) (uint64, error) {
 | 
						|
	data, err := ioutil.ReadFile(filepath.Join("/sys/class/net", ethInterface, "statistics", statsFile))
 | 
						|
	if err != nil {
 | 
						|
		return 0, err
 | 
						|
	}
 | 
						|
	return strconv.ParseUint(strings.TrimSpace(string(data)), 10, 64)
 | 
						|
}
 | 
						|
 | 
						|
// loopback is a network strategy that provides a basic loopback device
 | 
						|
type loopback struct {
 | 
						|
}
 | 
						|
 | 
						|
func (l *loopback) create(n *network, nspid int) error {
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (l *loopback) initialize(config *network) error {
 | 
						|
	return netlink.LinkSetUp(&netlink.Device{LinkAttrs: netlink.LinkAttrs{Name: "lo"}})
 | 
						|
}
 | 
						|
 | 
						|
func (l *loopback) attach(n *configs.Network) (err error) {
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (l *loopback) detach(n *configs.Network) (err error) {
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
// veth is a network strategy that uses a bridge and creates
 | 
						|
// a veth pair, one that is attached to the bridge on the host and the other
 | 
						|
// is placed inside the container's namespace
 | 
						|
type veth struct {
 | 
						|
}
 | 
						|
 | 
						|
func (v *veth) detach(n *configs.Network) (err error) {
 | 
						|
	return netlink.LinkSetMaster(&netlink.Device{LinkAttrs: netlink.LinkAttrs{Name: n.HostInterfaceName}}, nil)
 | 
						|
}
 | 
						|
 | 
						|
// attach a container network interface to an external network
 | 
						|
func (v *veth) attach(n *configs.Network) (err error) {
 | 
						|
	brl, err := netlink.LinkByName(n.Bridge)
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	br, ok := brl.(*netlink.Bridge)
 | 
						|
	if !ok {
 | 
						|
		return fmt.Errorf("Wrong device type %T", brl)
 | 
						|
	}
 | 
						|
	host, err := netlink.LinkByName(n.HostInterfaceName)
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
 | 
						|
	if err := netlink.LinkSetMaster(host, br); err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	if err := netlink.LinkSetMTU(host, n.Mtu); err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	if n.HairpinMode {
 | 
						|
		if err := netlink.LinkSetHairpin(host, true); err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
	}
 | 
						|
	if err := netlink.LinkSetUp(host); err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (v *veth) create(n *network, nspid int) (err error) {
 | 
						|
	tmpName, err := v.generateTempPeerName()
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	n.TempVethPeerName = tmpName
 | 
						|
	if n.Bridge == "" {
 | 
						|
		return fmt.Errorf("bridge is not specified")
 | 
						|
	}
 | 
						|
	veth := &netlink.Veth{
 | 
						|
		LinkAttrs: netlink.LinkAttrs{
 | 
						|
			Name:   n.HostInterfaceName,
 | 
						|
			TxQLen: n.TxQueueLen,
 | 
						|
		},
 | 
						|
		PeerName: n.TempVethPeerName,
 | 
						|
	}
 | 
						|
	if err := netlink.LinkAdd(veth); err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	defer func() {
 | 
						|
		if err != nil {
 | 
						|
			netlink.LinkDel(veth)
 | 
						|
		}
 | 
						|
	}()
 | 
						|
	if err := v.attach(&n.Network); err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	child, err := netlink.LinkByName(n.TempVethPeerName)
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	return netlink.LinkSetNsPid(child, nspid)
 | 
						|
}
 | 
						|
 | 
						|
func (v *veth) generateTempPeerName() (string, error) {
 | 
						|
	return utils.GenerateRandomName("veth", 7)
 | 
						|
}
 | 
						|
 | 
						|
func (v *veth) initialize(config *network) error {
 | 
						|
	peer := config.TempVethPeerName
 | 
						|
	if peer == "" {
 | 
						|
		return fmt.Errorf("peer is not specified")
 | 
						|
	}
 | 
						|
	child, err := netlink.LinkByName(peer)
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	if err := netlink.LinkSetDown(child); err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	if err := netlink.LinkSetName(child, config.Name); err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	// get the interface again after we changed the name as the index also changes.
 | 
						|
	if child, err = netlink.LinkByName(config.Name); err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	if config.MacAddress != "" {
 | 
						|
		mac, err := net.ParseMAC(config.MacAddress)
 | 
						|
		if err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
		if err := netlink.LinkSetHardwareAddr(child, mac); err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
	}
 | 
						|
	ip, err := netlink.ParseAddr(config.Address)
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	if err := netlink.AddrAdd(child, ip); err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	if config.IPv6Address != "" {
 | 
						|
		ip6, err := netlink.ParseAddr(config.IPv6Address)
 | 
						|
		if err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
		if err := netlink.AddrAdd(child, ip6); err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
	}
 | 
						|
	if err := netlink.LinkSetMTU(child, config.Mtu); err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	if err := netlink.LinkSetUp(child); err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	if config.Gateway != "" {
 | 
						|
		gw := net.ParseIP(config.Gateway)
 | 
						|
		if err := netlink.RouteAdd(&netlink.Route{
 | 
						|
			Scope:     netlink.SCOPE_UNIVERSE,
 | 
						|
			LinkIndex: child.Attrs().Index,
 | 
						|
			Gw:        gw,
 | 
						|
		}); err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
	}
 | 
						|
	if config.IPv6Gateway != "" {
 | 
						|
		gw := net.ParseIP(config.IPv6Gateway)
 | 
						|
		if err := netlink.RouteAdd(&netlink.Route{
 | 
						|
			Scope:     netlink.SCOPE_UNIVERSE,
 | 
						|
			LinkIndex: child.Attrs().Index,
 | 
						|
			Gw:        gw,
 | 
						|
		}); err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return nil
 | 
						|
}
 |