mirror of
https://github.com/shirou/gopsutil.git
synced 2025-04-29 13:49:21 +08:00

The order of init function execution is dependant on the order that the source files are passed to the compiler. This causes issues when building under other build systems, such as bazel or buck, as they are not guarenteed to maintain the same file order as the default go tool.
249 lines
6.1 KiB
Go
249 lines
6.1 KiB
Go
package net
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"net"
|
|
"strconv"
|
|
"strings"
|
|
"syscall"
|
|
|
|
"github.com/shirou/gopsutil/internal/common"
|
|
)
|
|
|
|
var invoke common.Invoker = common.Invoke{}
|
|
|
|
type IOCountersStat struct {
|
|
Name string `json:"name"` // interface name
|
|
BytesSent uint64 `json:"bytesSent"` // number of bytes sent
|
|
BytesRecv uint64 `json:"bytesRecv"` // number of bytes received
|
|
PacketsSent uint64 `json:"packetsSent"` // number of packets sent
|
|
PacketsRecv uint64 `json:"packetsRecv"` // number of packets received
|
|
Errin uint64 `json:"errin"` // total number of errors while receiving
|
|
Errout uint64 `json:"errout"` // total number of errors while sending
|
|
Dropin uint64 `json:"dropin"` // total number of incoming packets which were dropped
|
|
Dropout uint64 `json:"dropout"` // total number of outgoing packets which were dropped (always 0 on OSX and BSD)
|
|
Fifoin uint64 `json:"fifoin"` // total number of FIFO buffers errors while receiving
|
|
Fifoout uint64 `json:"fifoout"` // total number of FIFO buffers errors while sending
|
|
|
|
}
|
|
|
|
// Addr is implemented compatibility to psutil
|
|
type Addr struct {
|
|
IP string `json:"ip"`
|
|
Port uint32 `json:"port"`
|
|
}
|
|
|
|
type ConnectionStat struct {
|
|
Fd uint32 `json:"fd"`
|
|
Family uint32 `json:"family"`
|
|
Type uint32 `json:"type"`
|
|
Laddr Addr `json:"localaddr"`
|
|
Raddr Addr `json:"remoteaddr"`
|
|
Status string `json:"status"`
|
|
Uids []int32 `json:"uids"`
|
|
Pid int32 `json:"pid"`
|
|
}
|
|
|
|
// System wide stats about different network protocols
|
|
type ProtoCountersStat struct {
|
|
Protocol string `json:"protocol"`
|
|
Stats map[string]int64 `json:"stats"`
|
|
}
|
|
|
|
// NetInterfaceAddr is designed for represent interface addresses
|
|
type InterfaceAddr struct {
|
|
Addr string `json:"addr"`
|
|
}
|
|
|
|
type InterfaceStat struct {
|
|
MTU int `json:"mtu"` // maximum transmission unit
|
|
Name string `json:"name"` // e.g., "en0", "lo0", "eth0.100"
|
|
HardwareAddr string `json:"hardwareaddr"` // IEEE MAC-48, EUI-48 and EUI-64 form
|
|
Flags []string `json:"flags"` // e.g., FlagUp, FlagLoopback, FlagMulticast
|
|
Addrs []InterfaceAddr `json:"addrs"`
|
|
}
|
|
|
|
type FilterStat struct {
|
|
ConnTrackCount int64 `json:"conntrackCount"`
|
|
ConnTrackMax int64 `json:"conntrackMax"`
|
|
}
|
|
|
|
var constMap = map[string]int{
|
|
"TCP": syscall.SOCK_STREAM,
|
|
"UDP": syscall.SOCK_DGRAM,
|
|
"IPv4": syscall.AF_INET,
|
|
"IPv6": syscall.AF_INET6,
|
|
}
|
|
|
|
func (n IOCountersStat) String() string {
|
|
s, _ := json.Marshal(n)
|
|
return string(s)
|
|
}
|
|
|
|
func (n ConnectionStat) String() string {
|
|
s, _ := json.Marshal(n)
|
|
return string(s)
|
|
}
|
|
|
|
func (n ProtoCountersStat) String() string {
|
|
s, _ := json.Marshal(n)
|
|
return string(s)
|
|
}
|
|
|
|
func (a Addr) String() string {
|
|
s, _ := json.Marshal(a)
|
|
return string(s)
|
|
}
|
|
|
|
func (n InterfaceStat) String() string {
|
|
s, _ := json.Marshal(n)
|
|
return string(s)
|
|
}
|
|
|
|
func (n InterfaceAddr) String() string {
|
|
s, _ := json.Marshal(n)
|
|
return string(s)
|
|
}
|
|
|
|
func Interfaces() ([]InterfaceStat, error) {
|
|
return InterfacesWithContext(context.Background())
|
|
}
|
|
|
|
func InterfacesWithContext(ctx context.Context) ([]InterfaceStat, error) {
|
|
is, err := net.Interfaces()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
ret := make([]InterfaceStat, 0, len(is))
|
|
for _, ifi := range is {
|
|
|
|
var flags []string
|
|
if ifi.Flags&net.FlagUp != 0 {
|
|
flags = append(flags, "up")
|
|
}
|
|
if ifi.Flags&net.FlagBroadcast != 0 {
|
|
flags = append(flags, "broadcast")
|
|
}
|
|
if ifi.Flags&net.FlagLoopback != 0 {
|
|
flags = append(flags, "loopback")
|
|
}
|
|
if ifi.Flags&net.FlagPointToPoint != 0 {
|
|
flags = append(flags, "pointtopoint")
|
|
}
|
|
if ifi.Flags&net.FlagMulticast != 0 {
|
|
flags = append(flags, "multicast")
|
|
}
|
|
|
|
r := InterfaceStat{
|
|
Name: ifi.Name,
|
|
MTU: ifi.MTU,
|
|
HardwareAddr: ifi.HardwareAddr.String(),
|
|
Flags: flags,
|
|
}
|
|
addrs, err := ifi.Addrs()
|
|
if err == nil {
|
|
r.Addrs = make([]InterfaceAddr, 0, len(addrs))
|
|
for _, addr := range addrs {
|
|
r.Addrs = append(r.Addrs, InterfaceAddr{
|
|
Addr: addr.String(),
|
|
})
|
|
}
|
|
|
|
}
|
|
ret = append(ret, r)
|
|
}
|
|
|
|
return ret, nil
|
|
}
|
|
|
|
func getIOCountersAll(n []IOCountersStat) ([]IOCountersStat, error) {
|
|
r := IOCountersStat{
|
|
Name: "all",
|
|
}
|
|
for _, nic := range n {
|
|
r.BytesRecv += nic.BytesRecv
|
|
r.PacketsRecv += nic.PacketsRecv
|
|
r.Errin += nic.Errin
|
|
r.Dropin += nic.Dropin
|
|
r.BytesSent += nic.BytesSent
|
|
r.PacketsSent += nic.PacketsSent
|
|
r.Errout += nic.Errout
|
|
r.Dropout += nic.Dropout
|
|
}
|
|
|
|
return []IOCountersStat{r}, nil
|
|
}
|
|
|
|
func parseNetLine(line string) (ConnectionStat, error) {
|
|
f := strings.Fields(line)
|
|
if len(f) < 9 {
|
|
return ConnectionStat{}, fmt.Errorf("wrong line,%s", line)
|
|
}
|
|
|
|
pid, err := strconv.Atoi(f[1])
|
|
if err != nil {
|
|
return ConnectionStat{}, err
|
|
}
|
|
fd, err := strconv.Atoi(strings.Trim(f[3], "u"))
|
|
if err != nil {
|
|
return ConnectionStat{}, fmt.Errorf("unknown fd, %s", f[3])
|
|
}
|
|
netFamily, ok := constMap[f[4]]
|
|
if !ok {
|
|
return ConnectionStat{}, fmt.Errorf("unknown family, %s", f[4])
|
|
}
|
|
netType, ok := constMap[f[7]]
|
|
if !ok {
|
|
return ConnectionStat{}, fmt.Errorf("unknown type, %s", f[7])
|
|
}
|
|
|
|
laddr, raddr, err := parseNetAddr(f[8])
|
|
if err != nil {
|
|
return ConnectionStat{}, fmt.Errorf("failed to parse netaddr, %s", f[8])
|
|
}
|
|
|
|
n := ConnectionStat{
|
|
Fd: uint32(fd),
|
|
Family: uint32(netFamily),
|
|
Type: uint32(netType),
|
|
Laddr: laddr,
|
|
Raddr: raddr,
|
|
Pid: int32(pid),
|
|
}
|
|
if len(f) == 10 {
|
|
n.Status = strings.Trim(f[9], "()")
|
|
}
|
|
|
|
return n, nil
|
|
}
|
|
|
|
func parseNetAddr(line string) (laddr Addr, raddr Addr, err error) {
|
|
parse := func(l string) (Addr, error) {
|
|
host, port, err := net.SplitHostPort(l)
|
|
if err != nil {
|
|
return Addr{}, fmt.Errorf("wrong addr, %s", l)
|
|
}
|
|
lport, err := strconv.Atoi(port)
|
|
if err != nil {
|
|
return Addr{}, err
|
|
}
|
|
return Addr{IP: host, Port: uint32(lport)}, nil
|
|
}
|
|
|
|
addrs := strings.Split(line, "->")
|
|
if len(addrs) == 0 {
|
|
return laddr, raddr, fmt.Errorf("wrong netaddr, %s", line)
|
|
}
|
|
laddr, err = parse(addrs[0])
|
|
if len(addrs) == 2 { // remote addr exists
|
|
raddr, err = parse(addrs[1])
|
|
if err != nil {
|
|
return laddr, raddr, err
|
|
}
|
|
}
|
|
|
|
return laddr, raddr, err
|
|
}
|