package kvm import ( "fmt" "github.com/pion/mdns/v2" "golang.org/x/net/ipv4" "golang.org/x/net/ipv6" "net" "os/exec" "time" "github.com/vishvananda/netlink" "github.com/vishvananda/netlink/nl" ) var networkState struct { Up bool IPv4 string IPv6 string MAC string } type LocalIpInfo struct { IPv4 string IPv6 string MAC string } // setDhcpClientState sends signals to udhcpc to change it's current mode // of operation. Setting active to true will force udhcpc to renew the DHCP lease. // Setting active to false will put udhcpc into idle mode. func setDhcpClientState(active bool) { var signal string; if active { signal = "-SIGUSR1" } else { signal = "-SIGUSR2" } cmd := exec.Command("/usr/bin/killall", signal, "udhcpc"); if err := cmd.Run(); err != nil { fmt.Printf("network: setDhcpClientState: failed to change udhcpc state: %s\n", err) } } func checkNetworkState() { iface, err := netlink.LinkByName("eth0") if err != nil { fmt.Printf("failed to get eth0 interface: %v\n", err) return } newState := struct { Up bool IPv4 string IPv6 string MAC string }{ Up: iface.Attrs().OperState == netlink.OperUp, MAC: iface.Attrs().HardwareAddr.String(), } addrs, err := netlink.AddrList(iface, nl.FAMILY_ALL) if err != nil { fmt.Printf("failed to get addresses for eth0: %v\n", err) } // If the link is going down, put udhcpc into idle mode. // If the link is coming back up, activate udhcpc and force it to renew the lease. if newState.Up != networkState.Up { setDhcpClientState(newState.Up) } for _, addr := range addrs { if addr.IP.To4() != nil { if !newState.Up && networkState.Up { // If the network is going down, remove all IPv4 addresses from the interface. fmt.Printf("network: state transitioned to down, removing IPv4 address %s\n", addr.IP.String()) err := netlink.AddrDel(iface, &addr) if err != nil { fmt.Printf("network: failed to delete %s", addr.IP.String()) } newState.IPv4 = "..." } else { newState.IPv4 = addr.IP.String() } } else if addr.IP.To16() != nil && newState.IPv6 == "" { newState.IPv6 = addr.IP.String() } } if newState != networkState { networkState = newState fmt.Println("network state changed") requestDisplayUpdate() } } func startMDNS() error { addr4, err := net.ResolveUDPAddr("udp4", mdns.DefaultAddressIPv4) if err != nil { return err } addr6, err := net.ResolveUDPAddr("udp6", mdns.DefaultAddressIPv6) if err != nil { return err } l4, err := net.ListenUDP("udp4", addr4) if err != nil { return err } l6, err := net.ListenUDP("udp6", addr6) if err != nil { return err } _, err = mdns.Server(ipv4.NewPacketConn(l4), ipv6.NewPacketConn(l6), &mdns.Config{ LocalNames: []string{"jetkvm.local"}, //TODO: make it configurable }) if err != nil { return err } //defer server.Close() return nil } func init() { updates := make(chan netlink.LinkUpdate) done := make(chan struct{}) if err := netlink.LinkSubscribe(updates, done); err != nil { fmt.Println("failed to subscribe to link updates: %v", err) return } go func() { waitCtrlClientConnected() checkNetworkState() ticker := time.NewTicker(1 * time.Second) defer ticker.Stop() for { select { case update := <-updates: if update.Link.Attrs().Name == "eth0" { fmt.Printf("link update: %+v\n", update) checkNetworkState() } case <-ticker.C: checkNetworkState() case <-done: return } } }() fmt.Println("Starting mDNS server") err := startMDNS() if err != nil { fmt.Println("failed to run mDNS: %v", err) } }