202 lines
5.3 KiB
Go
202 lines
5.3 KiB
Go
// +build linux
|
|
|
|
package libcontainer
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"net"
|
|
"path/filepath"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/docker/libcontainer/configs"
|
|
"github.com/docker/libcontainer/netlink"
|
|
)
|
|
|
|
const defaultVethInterfaceName = "eth0"
|
|
|
|
var (
|
|
ErrNotValidStrategyType = errors.New("not a valid network strategy type")
|
|
)
|
|
|
|
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(*configs.Network, int) error
|
|
Initialize(*configs.Network) error
|
|
}
|
|
|
|
// getStrategy returns the specific network strategy for the
|
|
// provided type. If no strategy is registered for the type an
|
|
// ErrNotValidStrategyType is returned.
|
|
func getStrategy(tpe string) (networkStrategy, error) {
|
|
s, exists := strategies[tpe]
|
|
if !exists {
|
|
return nil, ErrNotValidStrategyType
|
|
}
|
|
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 *configs.Network, nspid int) error {
|
|
return nil
|
|
}
|
|
|
|
func (l *loopback) Initialize(config *configs.Network) error {
|
|
iface, err := net.InterfaceByName("lo")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return netlink.NetworkLinkUp(iface)
|
|
}
|
|
|
|
// veth is a network strategy that uses a bridge and creates
|
|
// a veth pair, one that stays outside on the host and the other
|
|
// is placed inside the container's namespace
|
|
type veth struct {
|
|
}
|
|
|
|
func (v *veth) Create(n *configs.Network, nspid int) error {
|
|
if n.Bridge == "" {
|
|
return fmt.Errorf("bridge is not specified")
|
|
}
|
|
bridge, err := net.InterfaceByName(n.Bridge)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := netlink.NetworkCreateVethPair(n.VethHost, n.VethChild, n.TxQueueLen); err != nil {
|
|
return err
|
|
}
|
|
host, err := net.InterfaceByName(n.VethHost)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := netlink.AddToBridge(host, bridge); err != nil {
|
|
return err
|
|
}
|
|
if err := netlink.NetworkSetMTU(host, n.Mtu); err != nil {
|
|
return err
|
|
}
|
|
if err := netlink.NetworkLinkUp(host); err != nil {
|
|
return err
|
|
}
|
|
child, err := net.InterfaceByName(n.VethChild)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return netlink.NetworkSetNsPid(child, nspid)
|
|
}
|
|
|
|
func (v *veth) Initialize(config *configs.Network) error {
|
|
vethChild := config.VethChild
|
|
if vethChild == "" {
|
|
return fmt.Errorf("vethChild is not specified")
|
|
}
|
|
child, err := net.InterfaceByName(vethChild)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := netlink.NetworkLinkDown(child); err != nil {
|
|
return err
|
|
}
|
|
if err := netlink.NetworkChangeName(child, defaultVethInterfaceName); err != nil {
|
|
return err
|
|
}
|
|
// get the interface again after we changed the name as the index also changes.
|
|
if child, err = net.InterfaceByName(defaultVethInterfaceName); err != nil {
|
|
return err
|
|
}
|
|
if config.MacAddress != "" {
|
|
if err := netlink.NetworkSetMacAddress(child, config.MacAddress); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
ip, ipNet, err := net.ParseCIDR(config.Address)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := netlink.NetworkLinkAddIp(child, ip, ipNet); err != nil {
|
|
return err
|
|
}
|
|
if config.IPv6Address != "" {
|
|
if ip, ipNet, err = net.ParseCIDR(config.IPv6Address); err != nil {
|
|
return err
|
|
}
|
|
if err := netlink.NetworkLinkAddIp(child, ip, ipNet); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
if err := netlink.NetworkSetMTU(child, config.Mtu); err != nil {
|
|
return err
|
|
}
|
|
if err := netlink.NetworkLinkUp(child); err != nil {
|
|
return err
|
|
}
|
|
if config.Gateway != "" {
|
|
if err := netlink.AddDefaultGw(config.Gateway, defaultVethInterfaceName); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
if config.IPv6Gateway != "" {
|
|
if err := netlink.AddDefaultGw(config.IPv6Gateway, defaultVethInterfaceName); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|