2014-05-23 02:00:41 +04:00
|
|
|
package system
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2014-07-21 22:25:22 +04:00
|
|
|
"log"
|
2014-05-23 02:00:41 +04:00
|
|
|
"net"
|
|
|
|
"os/exec"
|
2014-07-01 00:37:09 +04:00
|
|
|
"strings"
|
2014-06-13 04:47:05 +04:00
|
|
|
"time"
|
2014-05-23 02:00:41 +04:00
|
|
|
|
|
|
|
"github.com/coreos/coreos-cloudinit/network"
|
|
|
|
"github.com/coreos/coreos-cloudinit/third_party/github.com/dotcloud/docker/pkg/netlink"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
runtimeNetworkPath = "/run/systemd/network"
|
|
|
|
)
|
|
|
|
|
|
|
|
func RestartNetwork(interfaces []network.InterfaceGenerator) (err error) {
|
|
|
|
defer func() {
|
2014-06-13 04:47:05 +04:00
|
|
|
if e := restartNetworkd(); e != nil {
|
|
|
|
err = e
|
|
|
|
return
|
|
|
|
}
|
|
|
|
// TODO(crawford): Get rid of this once networkd fixes the race
|
|
|
|
// https://bugs.freedesktop.org/show_bug.cgi?id=76077
|
|
|
|
time.Sleep(5 * time.Second)
|
2014-05-23 02:00:41 +04:00
|
|
|
if e := restartNetworkd(); e != nil {
|
|
|
|
err = e
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
if err = downNetworkInterfaces(interfaces); err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2014-07-01 00:37:09 +04:00
|
|
|
if err = maybeProbe8012q(interfaces); err != nil {
|
2014-05-23 02:00:41 +04:00
|
|
|
return
|
|
|
|
}
|
2014-07-01 00:37:09 +04:00
|
|
|
return maybeProbeBonding(interfaces)
|
2014-05-23 02:00:41 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
func downNetworkInterfaces(interfaces []network.InterfaceGenerator) error {
|
|
|
|
sysInterfaceMap := make(map[string]*net.Interface)
|
|
|
|
if systemInterfaces, err := net.Interfaces(); err == nil {
|
|
|
|
for _, iface := range systemInterfaces {
|
2014-07-21 22:24:42 +04:00
|
|
|
iface := iface
|
|
|
|
sysInterfaceMap[iface.Name] = &iface
|
2014-05-23 02:00:41 +04:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, iface := range interfaces {
|
|
|
|
if systemInterface, ok := sysInterfaceMap[iface.Name()]; ok {
|
2014-07-21 22:25:22 +04:00
|
|
|
log.Printf("Taking down interface %q\n", systemInterface.Name)
|
2014-05-23 02:00:41 +04:00
|
|
|
if err := netlink.NetworkLinkDown(systemInterface); err != nil {
|
|
|
|
fmt.Printf("Error while downing interface %q (%s). Continuing...\n", systemInterface.Name, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2014-07-01 00:37:09 +04:00
|
|
|
func maybeProbe8012q(interfaces []network.InterfaceGenerator) error {
|
|
|
|
for _, iface := range interfaces {
|
|
|
|
if iface.Type() == "vlan" {
|
2014-07-21 22:25:22 +04:00
|
|
|
log.Printf("Probing LKM %q (%q)\n", "8021q", "8021q")
|
2014-07-01 00:37:09 +04:00
|
|
|
return exec.Command("modprobe", "8021q").Run()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func maybeProbeBonding(interfaces []network.InterfaceGenerator) error {
|
|
|
|
args := []string{"bonding"}
|
|
|
|
for _, iface := range interfaces {
|
|
|
|
if iface.Type() == "bond" {
|
|
|
|
args = append(args, strings.Split(iface.ModprobeParams(), " ")...)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
2014-07-21 22:25:22 +04:00
|
|
|
log.Printf("Probing LKM %q (%q)\n", "bonding", args)
|
2014-07-01 00:37:09 +04:00
|
|
|
return exec.Command("modprobe", args...).Run()
|
2014-05-23 02:00:41 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
func restartNetworkd() error {
|
2014-07-21 22:25:22 +04:00
|
|
|
log.Printf("Restarting networkd.service\n")
|
2014-06-06 04:40:53 +04:00
|
|
|
_, err := NewUnitManager("").RunUnitCommand("restart", "systemd-networkd.service")
|
2014-05-23 02:00:41 +04:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func WriteNetworkdConfigs(interfaces []network.InterfaceGenerator) error {
|
|
|
|
for _, iface := range interfaces {
|
2014-07-21 22:24:42 +04:00
|
|
|
filename := fmt.Sprintf("%s.netdev", iface.Filename())
|
2014-05-23 02:00:41 +04:00
|
|
|
if err := writeConfig(filename, iface.Netdev()); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2014-07-21 22:24:42 +04:00
|
|
|
filename = fmt.Sprintf("%s.link", iface.Filename())
|
2014-05-23 02:00:41 +04:00
|
|
|
if err := writeConfig(filename, iface.Link()); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2014-07-21 22:24:42 +04:00
|
|
|
filename = fmt.Sprintf("%s.network", iface.Filename())
|
2014-05-23 02:00:41 +04:00
|
|
|
if err := writeConfig(filename, iface.Network()); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func writeConfig(filename string, config string) error {
|
|
|
|
if config == "" {
|
|
|
|
return nil
|
|
|
|
}
|
2014-07-21 22:25:22 +04:00
|
|
|
log.Printf("Writing networkd unit %q\n", filename)
|
2014-07-21 22:24:42 +04:00
|
|
|
_, err := WriteFile(&File{Content: config, Path: filename}, runtimeNetworkPath)
|
|
|
|
return err
|
2014-05-23 02:00:41 +04:00
|
|
|
}
|