//go:build darwin // +build darwin package net import ( "context" "errors" "fmt" "os/exec" "regexp" "strconv" "strings" "github.com/shirou/gopsutil/v3/internal/common" ) var ( errNetstatHeader = errors.New("Can't parse header of netstat output") netstatLinkRegexp = regexp.MustCompile(`^$`) ) const endOfLine = "\n" func parseNetstatLine(line string) (stat *IOCountersStat, linkID *uint, err error) { var ( numericValue uint64 columns = strings.Fields(line) ) if columns[0] == "Name" { err = errNetstatHeader return } // try to extract the numeric value from if subMatch := netstatLinkRegexp.FindStringSubmatch(columns[2]); len(subMatch) == 2 { numericValue, err = strconv.ParseUint(subMatch[1], 10, 64) if err != nil { return } linkIDUint := uint(numericValue) linkID = &linkIDUint } base := 1 numberColumns := len(columns) // sometimes Address is omitted if numberColumns < 12 { base = 0 } if numberColumns < 11 || numberColumns > 13 { err = fmt.Errorf("Line %q do have an invalid number of columns %d", line, numberColumns) return } parsed := make([]uint64, 0, 7) vv := []string{ columns[base+3], // Ipkts == PacketsRecv columns[base+4], // Ierrs == Errin columns[base+5], // Ibytes == BytesRecv columns[base+6], // Opkts == PacketsSent columns[base+7], // Oerrs == Errout columns[base+8], // Obytes == BytesSent } if len(columns) == 12 { vv = append(vv, columns[base+10]) } for _, target := range vv { if target == "-" { parsed = append(parsed, 0) continue } if numericValue, err = strconv.ParseUint(target, 10, 64); err != nil { return } parsed = append(parsed, numericValue) } stat = &IOCountersStat{ Name: strings.Trim(columns[0], "*"), // remove the * that sometimes is on right on interface PacketsRecv: parsed[0], Errin: parsed[1], BytesRecv: parsed[2], PacketsSent: parsed[3], Errout: parsed[4], BytesSent: parsed[5], } if len(parsed) == 7 { stat.Dropout = parsed[6] } return } type netstatInterface struct { linkID *uint stat *IOCountersStat } func parseNetstatOutput(output string) ([]netstatInterface, error) { var ( err error lines = strings.Split(strings.Trim(output, endOfLine), endOfLine) ) // number of interfaces is number of lines less one for the header numberInterfaces := len(lines) - 1 interfaces := make([]netstatInterface, numberInterfaces) // no output beside header if numberInterfaces == 0 { return interfaces, nil } for index := 0; index < numberInterfaces; index++ { nsIface := netstatInterface{} if nsIface.stat, nsIface.linkID, err = parseNetstatLine(lines[index+1]); err != nil { return nil, err } interfaces[index] = nsIface } return interfaces, nil } // map that hold the name of a network interface and the number of usage type mapInterfaceNameUsage map[string]uint func newMapInterfaceNameUsage(ifaces []netstatInterface) mapInterfaceNameUsage { output := make(mapInterfaceNameUsage) for index := range ifaces { if ifaces[index].linkID != nil { ifaceName := ifaces[index].stat.Name usage, ok := output[ifaceName] if ok { output[ifaceName] = usage + 1 } else { output[ifaceName] = 1 } } } return output } func (min mapInterfaceNameUsage) isTruncated() bool { for _, usage := range min { if usage > 1 { return true } } return false } func (min mapInterfaceNameUsage) notTruncated() []string { output := make([]string, 0) for ifaceName, usage := range min { if usage == 1 { output = append(output, ifaceName) } } return output } // example of `netstat -ibdnW` output on yosemite // Name Mtu Network Address Ipkts Ierrs Ibytes Opkts Oerrs Obytes Coll Drop // lo0 16384 869107 0 169411755 869107 0 169411755 0 0 // lo0 16384 ::1/128 ::1 869107 - 169411755 869107 - 169411755 - - // lo0 16384 127 127.0.0.1 869107 - 169411755 869107 - 169411755 - - func IOCounters(pernic bool) ([]IOCountersStat, error) { return IOCountersWithContext(context.Background(), pernic) } func IOCountersWithContext(ctx context.Context, pernic bool) ([]IOCountersStat, error) { var ( ret []IOCountersStat retIndex int ) netstat, err := exec.LookPath("netstat") if err != nil { return nil, err } // try to get all interface metrics, and hope there won't be any truncated out, err := invoke.CommandWithContext(ctx, netstat, "-ibdnW") if err != nil { return nil, err } nsInterfaces, err := parseNetstatOutput(string(out)) if err != nil { return nil, err } ifaceUsage := newMapInterfaceNameUsage(nsInterfaces) notTruncated := ifaceUsage.notTruncated() ret = make([]IOCountersStat, len(notTruncated)) if !ifaceUsage.isTruncated() { // no truncated interface name, return stats of all interface with for index := range nsInterfaces { if nsInterfaces[index].linkID != nil { ret[retIndex] = *nsInterfaces[index].stat retIndex++ } } } else { // duplicated interface, list all interfaces if out, err = invoke.CommandWithContext(ctx, "ifconfig", "-l"); err != nil { return nil, err } interfaceNames := strings.Fields(strings.TrimRight(string(out), endOfLine)) // for each of the interface name, run netstat if we don't have any stats yet for _, interfaceName := range interfaceNames { truncated := true for index := range nsInterfaces { if nsInterfaces[index].linkID != nil && nsInterfaces[index].stat.Name == interfaceName { // handle the non truncated name to avoid execute netstat for them again ret[retIndex] = *nsInterfaces[index].stat retIndex++ truncated = false break } } if truncated { // run netstat with -I$ifacename if out, err = invoke.CommandWithContext(ctx, netstat, "-ibdnWI"+interfaceName); err != nil { return nil, err } parsedIfaces, err := parseNetstatOutput(string(out)) if err != nil { return nil, err } if len(parsedIfaces) == 0 { // interface had been removed since `ifconfig -l` had been executed continue } for index := range parsedIfaces { if parsedIfaces[index].linkID != nil { ret = append(ret, *parsedIfaces[index].stat) break } } } } } if pernic == false { return getIOCountersAll(ret) } return ret, nil } // IOCountersByFile exists just for compatibility with Linux. func IOCountersByFile(pernic bool, filename string) ([]IOCountersStat, error) { return IOCountersByFileWithContext(context.Background(), pernic, filename) } func IOCountersByFileWithContext(ctx context.Context, pernic bool, filename string) ([]IOCountersStat, error) { return IOCountersWithContext(ctx, pernic) } func FilterCounters() ([]FilterStat, error) { return FilterCountersWithContext(context.Background()) } func FilterCountersWithContext(ctx context.Context) ([]FilterStat, error) { return nil, common.ErrNotImplementedError } func ConntrackStats(percpu bool) ([]ConntrackStat, error) { return ConntrackStatsWithContext(context.Background(), percpu) } func ConntrackStatsWithContext(ctx context.Context, percpu bool) ([]ConntrackStat, error) { return nil, common.ErrNotImplementedError } // ProtoCounters returns network statistics for the entire system // If protocols is empty then all protocols are returned, otherwise // just the protocols in the list are returned. // Not Implemented for Darwin func ProtoCounters(protocols []string) ([]ProtoCountersStat, error) { return ProtoCountersWithContext(context.Background(), protocols) } func ProtoCountersWithContext(ctx context.Context, protocols []string) ([]ProtoCountersStat, error) { return nil, common.ErrNotImplementedError }