mirror of
https://github.com/containers/podman.git
synced 2025-08-06 19:44:14 +08:00

podman can now support adding network aliases when running containers (--network-alias). It requires an updated dnsname plugin as well as an updated ocicni to work properly. Signed-off-by: baude <bbaude@redhat.com>
151 lines
3.9 KiB
Go
151 lines
3.9 KiB
Go
// +build linux
|
|
|
|
package ocicni
|
|
|
|
import (
|
|
"fmt"
|
|
"net"
|
|
"os/exec"
|
|
"strings"
|
|
|
|
"github.com/containernetworking/plugins/pkg/ns"
|
|
"github.com/vishvananda/netlink"
|
|
)
|
|
|
|
var defaultNamespaceEnterCommandName = "nsenter"
|
|
|
|
type nsManager struct {
|
|
nsenterPath string
|
|
}
|
|
|
|
func (nsm *nsManager) init() error {
|
|
var err error
|
|
nsm.nsenterPath, err = exec.LookPath(defaultNamespaceEnterCommandName)
|
|
return err
|
|
}
|
|
|
|
func getContainerDetails(nsm *nsManager, netnsPath, interfaceName, addrType string) (*net.IPNet, *net.HardwareAddr, error) {
|
|
// Try to retrieve ip inside container network namespace
|
|
output, err := exec.Command(nsm.nsenterPath, fmt.Sprintf("--net=%s", netnsPath), "-F", "--",
|
|
"ip", "-o", addrType, "addr", "show", "dev", interfaceName, "scope", "global").CombinedOutput()
|
|
if err != nil {
|
|
return nil, nil, fmt.Errorf("Unexpected command output %s with error: %v", output, err)
|
|
}
|
|
|
|
lines := strings.Split(string(output), "\n")
|
|
if len(lines) < 1 {
|
|
return nil, nil, fmt.Errorf("Unexpected command output %s", output)
|
|
}
|
|
fields := strings.Fields(lines[0])
|
|
if len(fields) < 4 {
|
|
return nil, nil, fmt.Errorf("Unexpected address output %s ", lines[0])
|
|
}
|
|
ip, ipNet, err := net.ParseCIDR(fields[3])
|
|
if err != nil {
|
|
return nil, nil, fmt.Errorf("CNI failed to parse ip from output %s due to %v", output, err)
|
|
}
|
|
if ip.To4() == nil {
|
|
ipNet.IP = ip
|
|
} else {
|
|
ipNet.IP = ip.To4()
|
|
}
|
|
|
|
// Try to retrieve MAC inside container network namespace
|
|
output, err = exec.Command(nsm.nsenterPath, fmt.Sprintf("--net=%s", netnsPath), "-F", "--",
|
|
"ip", "link", "show", "dev", interfaceName).CombinedOutput()
|
|
if err != nil {
|
|
return nil, nil, fmt.Errorf("unexpected 'ip link' command output %s with error: %v", output, err)
|
|
}
|
|
|
|
lines = strings.Split(string(output), "\n")
|
|
if len(lines) < 2 {
|
|
return nil, nil, fmt.Errorf("unexpected 'ip link' command output %s", output)
|
|
}
|
|
fields = strings.Fields(lines[1])
|
|
if len(fields) < 4 {
|
|
return nil, nil, fmt.Errorf("unexpected link output %s ", lines[0])
|
|
}
|
|
mac, err := net.ParseMAC(fields[1])
|
|
if err != nil {
|
|
return nil, nil, fmt.Errorf("failed to parse MAC from output %s due to %v", output, err)
|
|
}
|
|
|
|
return ipNet, &mac, nil
|
|
}
|
|
|
|
func tearDownLoopback(netns string) error {
|
|
return ns.WithNetNSPath(netns, func(_ ns.NetNS) error {
|
|
link, err := netlink.LinkByName(loIfname)
|
|
if err != nil {
|
|
return err // not tested
|
|
}
|
|
err = netlink.LinkSetDown(link)
|
|
if err != nil {
|
|
return err // not tested
|
|
}
|
|
return nil
|
|
})
|
|
}
|
|
|
|
func bringUpLoopback(netns string) error {
|
|
if err := ns.WithNetNSPath(netns, func(_ ns.NetNS) error {
|
|
link, err := netlink.LinkByName(loIfname)
|
|
if err == nil {
|
|
err = netlink.LinkSetUp(link)
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
v4Addrs, err := netlink.AddrList(link, netlink.FAMILY_V4)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if len(v4Addrs) != 0 {
|
|
// sanity check that this is a loopback address
|
|
for _, addr := range v4Addrs {
|
|
if !addr.IP.IsLoopback() {
|
|
return fmt.Errorf("loopback interface found with non-loopback address %q", addr.IP)
|
|
}
|
|
}
|
|
}
|
|
|
|
v6Addrs, err := netlink.AddrList(link, netlink.FAMILY_V6)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if len(v6Addrs) != 0 {
|
|
// sanity check that this is a loopback address
|
|
for _, addr := range v6Addrs {
|
|
if !addr.IP.IsLoopback() {
|
|
return fmt.Errorf("loopback interface found with non-loopback address %q", addr.IP)
|
|
}
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}); err != nil {
|
|
return fmt.Errorf("error adding loopback interface: %s", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func checkLoopback(netns string) error {
|
|
// Make sure loopback interface is up
|
|
if err := ns.WithNetNSPath(netns, func(_ ns.NetNS) error {
|
|
link, err := netlink.LinkByName(loIfname)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if link.Attrs().Flags&net.FlagUp != net.FlagUp {
|
|
return fmt.Errorf("loopback interface is down")
|
|
}
|
|
|
|
return nil
|
|
}); err != nil {
|
|
return fmt.Errorf("error checking loopback interface: %v", err)
|
|
}
|
|
return nil
|
|
}
|