net_darwin.go 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284
  1. // +build darwin
  2. package net
  3. import (
  4. "context"
  5. "errors"
  6. "fmt"
  7. "os/exec"
  8. "regexp"
  9. "strconv"
  10. "strings"
  11. )
  12. var (
  13. errNetstatHeader = errors.New("Can't parse header of netstat output")
  14. netstatLinkRegexp = regexp.MustCompile(`^<Link#(\d+)>$`)
  15. )
  16. const endOfLine = "\n"
  17. func parseNetstatLine(line string) (stat *IOCountersStat, linkID *uint, err error) {
  18. var (
  19. numericValue uint64
  20. columns = strings.Fields(line)
  21. )
  22. if columns[0] == "Name" {
  23. err = errNetstatHeader
  24. return
  25. }
  26. // try to extract the numeric value from <Link#123>
  27. if subMatch := netstatLinkRegexp.FindStringSubmatch(columns[2]); len(subMatch) == 2 {
  28. numericValue, err = strconv.ParseUint(subMatch[1], 10, 64)
  29. if err != nil {
  30. return
  31. }
  32. linkIDUint := uint(numericValue)
  33. linkID = &linkIDUint
  34. }
  35. base := 1
  36. numberColumns := len(columns)
  37. // sometimes Address is ommitted
  38. if numberColumns < 12 {
  39. base = 0
  40. }
  41. if numberColumns < 11 || numberColumns > 13 {
  42. err = fmt.Errorf("Line %q do have an invalid number of columns %d", line, numberColumns)
  43. return
  44. }
  45. parsed := make([]uint64, 0, 7)
  46. vv := []string{
  47. columns[base+3], // Ipkts == PacketsRecv
  48. columns[base+4], // Ierrs == Errin
  49. columns[base+5], // Ibytes == BytesRecv
  50. columns[base+6], // Opkts == PacketsSent
  51. columns[base+7], // Oerrs == Errout
  52. columns[base+8], // Obytes == BytesSent
  53. }
  54. if len(columns) == 12 {
  55. vv = append(vv, columns[base+10])
  56. }
  57. for _, target := range vv {
  58. if target == "-" {
  59. parsed = append(parsed, 0)
  60. continue
  61. }
  62. if numericValue, err = strconv.ParseUint(target, 10, 64); err != nil {
  63. return
  64. }
  65. parsed = append(parsed, numericValue)
  66. }
  67. stat = &IOCountersStat{
  68. Name: strings.Trim(columns[0], "*"), // remove the * that sometimes is on right on interface
  69. PacketsRecv: parsed[0],
  70. Errin: parsed[1],
  71. BytesRecv: parsed[2],
  72. PacketsSent: parsed[3],
  73. Errout: parsed[4],
  74. BytesSent: parsed[5],
  75. }
  76. if len(parsed) == 7 {
  77. stat.Dropout = parsed[6]
  78. }
  79. return
  80. }
  81. type netstatInterface struct {
  82. linkID *uint
  83. stat *IOCountersStat
  84. }
  85. func parseNetstatOutput(output string) ([]netstatInterface, error) {
  86. var (
  87. err error
  88. lines = strings.Split(strings.Trim(output, endOfLine), endOfLine)
  89. )
  90. // number of interfaces is number of lines less one for the header
  91. numberInterfaces := len(lines) - 1
  92. interfaces := make([]netstatInterface, numberInterfaces)
  93. // no output beside header
  94. if numberInterfaces == 0 {
  95. return interfaces, nil
  96. }
  97. for index := 0; index < numberInterfaces; index++ {
  98. nsIface := netstatInterface{}
  99. if nsIface.stat, nsIface.linkID, err = parseNetstatLine(lines[index+1]); err != nil {
  100. return nil, err
  101. }
  102. interfaces[index] = nsIface
  103. }
  104. return interfaces, nil
  105. }
  106. // map that hold the name of a network interface and the number of usage
  107. type mapInterfaceNameUsage map[string]uint
  108. func newMapInterfaceNameUsage(ifaces []netstatInterface) mapInterfaceNameUsage {
  109. output := make(mapInterfaceNameUsage)
  110. for index := range ifaces {
  111. if ifaces[index].linkID != nil {
  112. ifaceName := ifaces[index].stat.Name
  113. usage, ok := output[ifaceName]
  114. if ok {
  115. output[ifaceName] = usage + 1
  116. } else {
  117. output[ifaceName] = 1
  118. }
  119. }
  120. }
  121. return output
  122. }
  123. func (min mapInterfaceNameUsage) isTruncated() bool {
  124. for _, usage := range min {
  125. if usage > 1 {
  126. return true
  127. }
  128. }
  129. return false
  130. }
  131. func (min mapInterfaceNameUsage) notTruncated() []string {
  132. output := make([]string, 0)
  133. for ifaceName, usage := range min {
  134. if usage == 1 {
  135. output = append(output, ifaceName)
  136. }
  137. }
  138. return output
  139. }
  140. // example of `netstat -ibdnW` output on yosemite
  141. // Name Mtu Network Address Ipkts Ierrs Ibytes Opkts Oerrs Obytes Coll Drop
  142. // lo0 16384 <Link#1> 869107 0 169411755 869107 0 169411755 0 0
  143. // lo0 16384 ::1/128 ::1 869107 - 169411755 869107 - 169411755 - -
  144. // lo0 16384 127 127.0.0.1 869107 - 169411755 869107 - 169411755 - -
  145. func IOCounters(pernic bool) ([]IOCountersStat, error) {
  146. return IOCountersWithContext(context.Background(), pernic)
  147. }
  148. func IOCountersWithContext(ctx context.Context, pernic bool) ([]IOCountersStat, error) {
  149. var (
  150. ret []IOCountersStat
  151. retIndex int
  152. )
  153. netstat, err := exec.LookPath("/usr/sbin/netstat")
  154. if err != nil {
  155. return nil, err
  156. }
  157. // try to get all interface metrics, and hope there won't be any truncated
  158. out, err := invoke.CommandWithContext(ctx, netstat, "-ibdnW")
  159. if err != nil {
  160. return nil, err
  161. }
  162. nsInterfaces, err := parseNetstatOutput(string(out))
  163. if err != nil {
  164. return nil, err
  165. }
  166. ifaceUsage := newMapInterfaceNameUsage(nsInterfaces)
  167. notTruncated := ifaceUsage.notTruncated()
  168. ret = make([]IOCountersStat, len(notTruncated))
  169. if !ifaceUsage.isTruncated() {
  170. // no truncated interface name, return stats of all interface with <Link#...>
  171. for index := range nsInterfaces {
  172. if nsInterfaces[index].linkID != nil {
  173. ret[retIndex] = *nsInterfaces[index].stat
  174. retIndex++
  175. }
  176. }
  177. } else {
  178. // duplicated interface, list all interfaces
  179. ifconfig, err := exec.LookPath("/sbin/ifconfig")
  180. if err != nil {
  181. return nil, err
  182. }
  183. if out, err = invoke.CommandWithContext(ctx, ifconfig, "-l"); err != nil {
  184. return nil, err
  185. }
  186. interfaceNames := strings.Fields(strings.TrimRight(string(out), endOfLine))
  187. // for each of the interface name, run netstat if we don't have any stats yet
  188. for _, interfaceName := range interfaceNames {
  189. truncated := true
  190. for index := range nsInterfaces {
  191. if nsInterfaces[index].linkID != nil && nsInterfaces[index].stat.Name == interfaceName {
  192. // handle the non truncated name to avoid execute netstat for them again
  193. ret[retIndex] = *nsInterfaces[index].stat
  194. retIndex++
  195. truncated = false
  196. break
  197. }
  198. }
  199. if truncated {
  200. // run netstat with -I$ifacename
  201. if out, err = invoke.CommandWithContext(ctx, netstat, "-ibdnWI"+interfaceName); err != nil {
  202. return nil, err
  203. }
  204. parsedIfaces, err := parseNetstatOutput(string(out))
  205. if err != nil {
  206. return nil, err
  207. }
  208. if len(parsedIfaces) == 0 {
  209. // interface had been removed since `ifconfig -l` had been executed
  210. continue
  211. }
  212. for index := range parsedIfaces {
  213. if parsedIfaces[index].linkID != nil {
  214. ret = append(ret, *parsedIfaces[index].stat)
  215. break
  216. }
  217. }
  218. }
  219. }
  220. }
  221. if pernic == false {
  222. return getIOCountersAll(ret)
  223. }
  224. return ret, nil
  225. }
  226. // NetIOCountersByFile is an method which is added just a compatibility for linux.
  227. func IOCountersByFile(pernic bool, filename string) ([]IOCountersStat, error) {
  228. return IOCountersByFileWithContext(context.Background(), pernic, filename)
  229. }
  230. func IOCountersByFileWithContext(ctx context.Context, pernic bool, filename string) ([]IOCountersStat, error) {
  231. return IOCounters(pernic)
  232. }
  233. func FilterCounters() ([]FilterStat, error) {
  234. return FilterCountersWithContext(context.Background())
  235. }
  236. func FilterCountersWithContext(ctx context.Context) ([]FilterStat, error) {
  237. return nil, errors.New("NetFilterCounters not implemented for darwin")
  238. }
  239. // NetProtoCounters returns network statistics for the entire system
  240. // If protocols is empty then all protocols are returned, otherwise
  241. // just the protocols in the list are returned.
  242. // Not Implemented for Darwin
  243. func ProtoCounters(protocols []string) ([]ProtoCountersStat, error) {
  244. return ProtoCountersWithContext(context.Background(), protocols)
  245. }
  246. func ProtoCountersWithContext(ctx context.Context, protocols []string) ([]ProtoCountersStat, error) {
  247. return nil, errors.New("NetProtoCounters not implemented for darwin")
  248. }