2018-06-03 20:05:13 +02:00

255 lines
4.9 KiB
Go

package diag
import (
"fmt"
"net"
"time"
"github.com/digineo/go-ping"
"github.com/vishvananda/netlink"
)
func formatRTT(rtt time.Duration) string {
return fmt.Sprintf("%.2fms", float64(rtt)/float64(time.Millisecond))
}
type ping4gw struct {
children []Node
}
func (d *ping4gw) String() string {
return "ping4/<default-gateway>"
}
func (d *ping4gw) Then(t Node) Node {
d.children = append(d.children, t)
return d
}
func (d *ping4gw) Children() []Node {
return d.children
}
func defaultIPv4Gateway() (string, error) {
rl, err := netlink.RouteGet(net.ParseIP("8.8.8.8"))
if err != nil {
return "", err
}
if got, want := len(rl), 1; got != want {
return "", fmt.Errorf("unexpected number of default routes: got %d, want %d", got, want)
}
r := rl[0]
return r.Gw.String(), nil
}
func (d *ping4gw) Evaluate() (string, error) {
const timeout = 1 * time.Second
gw, err := defaultIPv4Gateway()
if err != nil {
return "", nil
}
addr, err := net.ResolveIPAddr("ip4", gw)
if err != nil {
return "", err
}
p, err := ping.New("0.0.0.0", "")
if err != nil {
return "", err
}
rtt, err := p.Ping(addr, timeout)
if err != nil {
return "", err
//return fmt.Errorf("%s did not respond within %v", gw, timeout)
}
return formatRTT(rtt), nil
}
// Ping4Gateway returns a Node which succeeds when the default gateway responds
// to an ICMPv4 ping.
func Ping4Gateway() Node {
return &ping4gw{}
}
type ping4 struct {
children []Node
addr string
}
func (d *ping4) String() string {
return "ping4/" + d.addr
}
func (d *ping4) Then(t Node) Node {
d.children = append(d.children, t)
return d
}
func (d *ping4) Children() []Node {
return d.children
}
func (d *ping4) Evaluate() (string, error) {
const timeout = 1 * time.Second
addr, err := net.ResolveIPAddr("ip4", d.addr)
if err != nil {
return "", err
}
p, err := ping.New("0.0.0.0", "")
if err != nil {
return "", err
}
rtt, err := p.Ping(addr, timeout)
if err != nil {
return "", err
//return fmt.Errorf("%s did not respond within %v", gw, timeout)
}
return formatRTT(rtt), nil
}
// Ping4 returns a Node which succeeds when the specified address responds to an
// ICMPv4 ping.
func Ping4(addr string) Node {
return &ping4{addr: addr}
}
type ping6gw struct {
children []Node
}
func (d *ping6gw) String() string {
return "ping6gw/<default-gateway>"
}
func (d *ping6gw) Then(t Node) Node {
d.children = append(d.children, t)
return d
}
func (d *ping6gw) Children() []Node {
return d.children
}
func defaultIPv6Gateway() (string, error) {
rl, err := netlink.RouteGet(net.IPv6zero)
if err != nil {
return "", err
}
if got, want := len(rl), 1; got != want {
return "", fmt.Errorf("unexpected number of default routes: got %d, want %d", got, want)
}
r := rl[0]
iface, err := net.InterfaceByIndex(r.LinkIndex)
if err != nil {
return "", err
}
return r.Gw.String() + "%" + iface.Name, nil
}
func (d *ping6gw) Evaluate() (string, error) {
const timeout = 1 * time.Second
gw, err := defaultIPv6Gateway()
if err != nil {
return "", err
}
addr, err := net.ResolveIPAddr("ip6", gw)
if err != nil {
return "", err
}
p, err := ping.New("", "::")
if err != nil {
return "", err
}
rtt, err := p.Ping(addr, timeout)
if err != nil {
return "", err
//return fmt.Errorf("%s did not respond within %v", gw, timeout)
}
return formatRTT(rtt), nil
}
// Ping6Gateway returns a Node which succeeds when the default gateway responds
// to an ICMPv6 ping.
func Ping6Gateway() Node {
return &ping6gw{}
}
type ping6 struct {
children []Node
addr string
ifname string
}
func (d *ping6) String() string {
if d.ifname == "" {
return "ping6/" + d.addr
}
return fmt.Sprintf("ping6/%s→%s", d.ifname, d.addr)
}
func (d *ping6) Then(t Node) Node {
d.children = append(d.children, t)
return d
}
func (d *ping6) Children() []Node {
return d.children
}
func (d *ping6) Evaluate() (string, error) {
const timeout = 1 * time.Second
addr, err := net.ResolveIPAddr("ip6", d.addr)
if err != nil {
return "", err
}
bind6 := "::"
if d.ifname != "" {
iface, err := net.InterfaceByName(d.ifname)
if err != nil {
return "", err
}
addrs, err := iface.Addrs()
if err != nil {
return "", err
}
for _, addr := range addrs {
ipnet, ok := addr.(*net.IPNet)
if !ok {
continue
}
if ipnet.IP.To4() != nil {
continue // skip IPv4 addresses
}
if !global.Contains(ipnet.IP) {
continue // skip local IPv6 addresses
}
bind6 = ipnet.IP.String()
break
}
}
p, err := ping.New("", bind6)
if err != nil {
return "", err
}
rtt, err := p.Ping(addr, timeout)
if err != nil {
return "", err
//return fmt.Errorf("%s did not respond within %v", gw, timeout)
}
return formatRTT(rtt), nil
}
// Ping6 returns a Node which succeeds when the specified address responds to an
// ICMPv6 ping.
func Ping6(ifname, addr string) Node {
return &ping6{
ifname: ifname,
addr: addr,
}
}