| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425 |
- // +build aix
- package net
- import (
- "context"
- "fmt"
- "os/exec"
- "regexp"
- "strconv"
- "strings"
- "syscall"
- "github.com/shirou/gopsutil/internal/common"
- )
- func parseNetstatI(output string) ([]IOCountersStat, error) {
- lines := strings.Split(string(output), "\n")
- ret := make([]IOCountersStat, 0, len(lines)-1)
- exists := make([]string, 0, len(ret))
- // Check first line is header
- if len(lines) > 0 && strings.Fields(lines[0])[0] != "Name" {
- return nil, fmt.Errorf("not a 'netstat -i' output")
- }
- for _, line := range lines[1:] {
- values := strings.Fields(line)
- if len(values) < 1 || values[0] == "Name" {
- continue
- }
- if common.StringsHas(exists, values[0]) {
- // skip if already get
- continue
- }
- exists = append(exists, values[0])
- if len(values) < 9 {
- continue
- }
- base := 1
- // sometimes Address is omitted
- if len(values) < 10 {
- base = 0
- }
- parsed := make([]uint64, 0, 5)
- vv := []string{
- values[base+3], // Ipkts == PacketsRecv
- values[base+4], // Ierrs == Errin
- values[base+5], // Opkts == PacketsSent
- values[base+6], // Oerrs == Errout
- values[base+8], // Drops == Dropout
- }
- for _, target := range vv {
- if target == "-" {
- parsed = append(parsed, 0)
- continue
- }
- t, err := strconv.ParseUint(target, 10, 64)
- if err != nil {
- return nil, err
- }
- parsed = append(parsed, t)
- }
- n := IOCountersStat{
- Name: values[0],
- PacketsRecv: parsed[0],
- Errin: parsed[1],
- PacketsSent: parsed[2],
- Errout: parsed[3],
- Dropout: parsed[4],
- }
- ret = append(ret, n)
- }
- return ret, nil
- }
- func IOCounters(pernic bool) ([]IOCountersStat, error) {
- return IOCountersWithContext(context.Background(), pernic)
- }
- func IOCountersWithContext(ctx context.Context, pernic bool) ([]IOCountersStat, error) {
- netstat, err := exec.LookPath("netstat")
- if err != nil {
- return nil, err
- }
- out, err := invoke.CommandWithContext(ctx, netstat, "-idn")
- if err != nil {
- return nil, err
- }
- iocounters, err := parseNetstatI(string(out))
- if err != nil {
- return nil, err
- }
- if pernic == false {
- return getIOCountersAll(iocounters)
- }
- return iocounters, nil
- }
- // NetIOCountersByFile is an method which is added just a compatibility for 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 IOCounters(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
- }
- func ProtoCounters(protocols []string) ([]ProtoCountersStat, error) {
- return ProtoCountersWithContext(context.Background(), protocols)
- }
- func ProtoCountersWithContext(ctx context.Context, protocols []string) ([]ProtoCountersStat, error) {
- return nil, common.ErrNotImplementedError
- }
- func parseNetstatNetLine(line string) (ConnectionStat, error) {
- f := strings.Fields(line)
- if len(f) < 5 {
- return ConnectionStat{}, fmt.Errorf("wrong line,%s", line)
- }
- var netType, netFamily uint32
- switch f[0] {
- case "tcp", "tcp4":
- netType = syscall.SOCK_STREAM
- netFamily = syscall.AF_INET
- case "udp", "udp4":
- netType = syscall.SOCK_DGRAM
- netFamily = syscall.AF_INET
- case "tcp6":
- netType = syscall.SOCK_STREAM
- netFamily = syscall.AF_INET6
- case "udp6":
- netType = syscall.SOCK_DGRAM
- netFamily = syscall.AF_INET6
- default:
- return ConnectionStat{}, fmt.Errorf("unknown type, %s", f[0])
- }
- laddr, raddr, err := parseNetstatAddr(f[3], f[4], netFamily)
- if err != nil {
- return ConnectionStat{}, fmt.Errorf("failed to parse netaddr, %s %s", f[3], f[4])
- }
- n := ConnectionStat{
- Fd: uint32(0), // not supported
- Family: uint32(netFamily),
- Type: uint32(netType),
- Laddr: laddr,
- Raddr: raddr,
- Pid: int32(0), // not supported
- }
- if len(f) == 6 {
- n.Status = f[5]
- }
- return n, nil
- }
- var portMatch = regexp.MustCompile(`(.*)\.(\d+)$`)
- // This function only works for netstat returning addresses with a "."
- // before the port (0.0.0.0.22 instead of 0.0.0.0:22).
- func parseNetstatAddr(local string, remote string, family uint32) (laddr Addr, raddr Addr, err error) {
- parse := func(l string) (Addr, error) {
- matches := portMatch.FindStringSubmatch(l)
- if matches == nil {
- return Addr{}, fmt.Errorf("wrong addr, %s", l)
- }
- host := matches[1]
- port := matches[2]
- if host == "*" {
- switch family {
- case syscall.AF_INET:
- host = "0.0.0.0"
- case syscall.AF_INET6:
- host = "::"
- default:
- return Addr{}, fmt.Errorf("unknown family, %d", family)
- }
- }
- lport, err := strconv.Atoi(port)
- if err != nil {
- return Addr{}, err
- }
- return Addr{IP: host, Port: uint32(lport)}, nil
- }
- laddr, err = parse(local)
- if remote != "*.*" { // remote addr exists
- raddr, err = parse(remote)
- if err != nil {
- return laddr, raddr, err
- }
- }
- return laddr, raddr, err
- }
- func parseNetstatUnixLine(f []string) (ConnectionStat, error) {
- if len(f) < 8 {
- return ConnectionStat{}, fmt.Errorf("wrong number of fields: expected >=8 got %d", len(f))
- }
- var netType uint32
- switch f[1] {
- case "dgram":
- netType = syscall.SOCK_DGRAM
- case "stream":
- netType = syscall.SOCK_STREAM
- default:
- return ConnectionStat{}, fmt.Errorf("unknown type: %s", f[1])
- }
- // Some Unix Socket don't have any address associated
- addr := ""
- if len(f) == 9 {
- addr = f[8]
- }
- c := ConnectionStat{
- Fd: uint32(0), // not supported
- Family: uint32(syscall.AF_UNIX),
- Type: uint32(netType),
- Laddr: Addr{
- IP: addr,
- },
- Status: "NONE",
- Pid: int32(0), // not supported
- }
- return c, nil
- }
- // Return true if proto is the corresponding to the kind parameter
- // Only for Inet lines
- func hasCorrectInetProto(kind, proto string) bool {
- switch kind {
- case "all", "inet":
- return true
- case "unix":
- return false
- case "inet4":
- return !strings.HasSuffix(proto, "6")
- case "inet6":
- return strings.HasSuffix(proto, "6")
- case "tcp":
- return proto == "tcp" || proto == "tcp4" || proto == "tcp6"
- case "tcp4":
- return proto == "tcp" || proto == "tcp4"
- case "tcp6":
- return proto == "tcp6"
- case "udp":
- return proto == "udp" || proto == "udp4" || proto == "udp6"
- case "udp4":
- return proto == "udp" || proto == "udp4"
- case "udp6":
- return proto == "udp6"
- }
- return false
- }
- func parseNetstatA(output string, kind string) ([]ConnectionStat, error) {
- var ret []ConnectionStat
- lines := strings.Split(string(output), "\n")
- for _, line := range lines {
- fields := strings.Fields(line)
- if len(fields) < 1 {
- continue
- }
- if strings.HasPrefix(fields[0], "f1") {
- // Unix lines
- if len(fields) < 2 {
- // every unix connections have two lines
- continue
- }
- c, err := parseNetstatUnixLine(fields)
- if err != nil {
- return nil, fmt.Errorf("failed to parse Unix Address (%s): %s", line, err)
- }
- ret = append(ret, c)
- } else if strings.HasPrefix(fields[0], "tcp") || strings.HasPrefix(fields[0], "udp") {
- // Inet lines
- if !hasCorrectInetProto(kind, fields[0]) {
- continue
- }
- // On AIX, netstat display some connections with "*.*" as local addresses
- // Skip them as they aren't real connections.
- if fields[3] == "*.*" {
- continue
- }
- c, err := parseNetstatNetLine(line)
- if err != nil {
- return nil, fmt.Errorf("failed to parse Inet Address (%s): %s", line, err)
- }
- ret = append(ret, c)
- } else {
- // Header lines
- continue
- }
- }
- return ret, nil
- }
- func Connections(kind string) ([]ConnectionStat, error) {
- return ConnectionsWithContext(context.Background(), kind)
- }
- func ConnectionsWithContext(ctx context.Context, kind string) ([]ConnectionStat, error) {
- args := []string{"-na"}
- switch strings.ToLower(kind) {
- default:
- fallthrough
- case "":
- kind = "all"
- case "all":
- // nothing to add
- case "inet", "inet4", "inet6":
- args = append(args, "-finet")
- case "tcp", "tcp4", "tcp6":
- args = append(args, "-finet")
- case "udp", "udp4", "udp6":
- args = append(args, "-finet")
- case "unix":
- args = append(args, "-funix")
- }
- netstat, err := exec.LookPath("netstat")
- if err != nil {
- return nil, err
- }
- out, err := invoke.CommandWithContext(ctx, netstat, args...)
- if err != nil {
- return nil, err
- }
- ret, err := parseNetstatA(string(out), kind)
- if err != nil {
- return nil, err
- }
- return ret, nil
- }
- func ConnectionsMax(kind string, max int) ([]ConnectionStat, error) {
- return ConnectionsMaxWithContext(context.Background(), kind, max)
- }
- func ConnectionsMaxWithContext(ctx context.Context, kind string, max int) ([]ConnectionStat, error) {
- return []ConnectionStat{}, common.ErrNotImplementedError
- }
- // Return a list of network connections opened, omitting `Uids`.
- // WithoutUids functions are reliant on implementation details. They may be altered to be an alias for Connections or be
- // removed from the API in the future.
- func ConnectionsWithoutUids(kind string) ([]ConnectionStat, error) {
- return ConnectionsWithoutUidsWithContext(context.Background(), kind)
- }
- func ConnectionsWithoutUidsWithContext(ctx context.Context, kind string) ([]ConnectionStat, error) {
- return ConnectionsMaxWithoutUidsWithContext(ctx, kind, 0)
- }
- func ConnectionsMaxWithoutUidsWithContext(ctx context.Context, kind string, max int) ([]ConnectionStat, error) {
- return ConnectionsPidMaxWithoutUidsWithContext(ctx, kind, 0, max)
- }
- func ConnectionsPidWithoutUids(kind string, pid int32) ([]ConnectionStat, error) {
- return ConnectionsPidWithoutUidsWithContext(context.Background(), kind, pid)
- }
- func ConnectionsPidWithoutUidsWithContext(ctx context.Context, kind string, pid int32) ([]ConnectionStat, error) {
- return ConnectionsPidMaxWithoutUidsWithContext(ctx, kind, pid, 0)
- }
- func ConnectionsPidMaxWithoutUids(kind string, pid int32, max int) ([]ConnectionStat, error) {
- return ConnectionsPidMaxWithoutUidsWithContext(context.Background(), kind, pid, max)
- }
- func ConnectionsPidMaxWithoutUidsWithContext(ctx context.Context, kind string, pid int32, max int) ([]ConnectionStat, error) {
- return connectionsPidMaxWithoutUidsWithContext(ctx, kind, pid, max)
- }
- func connectionsPidMaxWithoutUidsWithContext(ctx context.Context, kind string, pid int32, max int) ([]ConnectionStat, error) {
- return []ConnectionStat{}, common.ErrNotImplementedError
- }
|