process.go 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241
  1. package process
  2. import (
  3. "context"
  4. "encoding/json"
  5. "errors"
  6. "runtime"
  7. "time"
  8. "github.com/shirou/gopsutil/cpu"
  9. "github.com/shirou/gopsutil/internal/common"
  10. "github.com/shirou/gopsutil/mem"
  11. )
  12. var (
  13. invoke common.Invoker = common.Invoke{}
  14. ErrorNoChildren = errors.New("process does not have children")
  15. )
  16. type Process struct {
  17. Pid int32 `json:"pid"`
  18. name string
  19. status string
  20. parent int32
  21. numCtxSwitches *NumCtxSwitchesStat
  22. uids []int32
  23. gids []int32
  24. numThreads int32
  25. memInfo *MemoryInfoStat
  26. sigInfo *SignalInfoStat
  27. lastCPUTimes *cpu.TimesStat
  28. lastCPUTime time.Time
  29. tgid int32
  30. }
  31. type OpenFilesStat struct {
  32. Path string `json:"path"`
  33. Fd uint64 `json:"fd"`
  34. }
  35. type MemoryInfoStat struct {
  36. RSS uint64 `json:"rss"` // bytes
  37. VMS uint64 `json:"vms"` // bytes
  38. Data uint64 `json:"data"` // bytes
  39. Stack uint64 `json:"stack"` // bytes
  40. Locked uint64 `json:"locked"` // bytes
  41. Swap uint64 `json:"swap"` // bytes
  42. }
  43. type SignalInfoStat struct {
  44. PendingProcess uint64 `json:"pending_process"`
  45. PendingThread uint64 `json:"pending_thread"`
  46. Blocked uint64 `json:"blocked"`
  47. Ignored uint64 `json:"ignored"`
  48. Caught uint64 `json:"caught"`
  49. }
  50. type RlimitStat struct {
  51. Resource int32 `json:"resource"`
  52. Soft int32 `json:"soft"` //TODO too small. needs to be uint64
  53. Hard int32 `json:"hard"` //TODO too small. needs to be uint64
  54. Used uint64 `json:"used"`
  55. }
  56. type IOCountersStat struct {
  57. ReadCount uint64 `json:"readCount"`
  58. WriteCount uint64 `json:"writeCount"`
  59. ReadBytes uint64 `json:"readBytes"`
  60. WriteBytes uint64 `json:"writeBytes"`
  61. }
  62. type NumCtxSwitchesStat struct {
  63. Voluntary int64 `json:"voluntary"`
  64. Involuntary int64 `json:"involuntary"`
  65. }
  66. // Resource limit constants are from /usr/include/x86_64-linux-gnu/bits/resource.h
  67. // from libc6-dev package in Ubuntu 16.10
  68. const (
  69. RLIMIT_CPU int32 = 0
  70. RLIMIT_FSIZE int32 = 1
  71. RLIMIT_DATA int32 = 2
  72. RLIMIT_STACK int32 = 3
  73. RLIMIT_CORE int32 = 4
  74. RLIMIT_RSS int32 = 5
  75. RLIMIT_NPROC int32 = 6
  76. RLIMIT_NOFILE int32 = 7
  77. RLIMIT_MEMLOCK int32 = 8
  78. RLIMIT_AS int32 = 9
  79. RLIMIT_LOCKS int32 = 10
  80. RLIMIT_SIGPENDING int32 = 11
  81. RLIMIT_MSGQUEUE int32 = 12
  82. RLIMIT_NICE int32 = 13
  83. RLIMIT_RTPRIO int32 = 14
  84. RLIMIT_RTTIME int32 = 15
  85. )
  86. func (p Process) String() string {
  87. s, _ := json.Marshal(p)
  88. return string(s)
  89. }
  90. func (o OpenFilesStat) String() string {
  91. s, _ := json.Marshal(o)
  92. return string(s)
  93. }
  94. func (m MemoryInfoStat) String() string {
  95. s, _ := json.Marshal(m)
  96. return string(s)
  97. }
  98. func (r RlimitStat) String() string {
  99. s, _ := json.Marshal(r)
  100. return string(s)
  101. }
  102. func (i IOCountersStat) String() string {
  103. s, _ := json.Marshal(i)
  104. return string(s)
  105. }
  106. func (p NumCtxSwitchesStat) String() string {
  107. s, _ := json.Marshal(p)
  108. return string(s)
  109. }
  110. func PidExists(pid int32) (bool, error) {
  111. return PidExistsWithContext(context.Background(), pid)
  112. }
  113. func PidExistsWithContext(ctx context.Context, pid int32) (bool, error) {
  114. pids, err := Pids()
  115. if err != nil {
  116. return false, err
  117. }
  118. for _, i := range pids {
  119. if i == pid {
  120. return true, err
  121. }
  122. }
  123. return false, err
  124. }
  125. // If interval is 0, return difference from last call(non-blocking).
  126. // If interval > 0, wait interval sec and return diffrence between start and end.
  127. func (p *Process) Percent(interval time.Duration) (float64, error) {
  128. return p.PercentWithContext(context.Background(), interval)
  129. }
  130. func (p *Process) PercentWithContext(ctx context.Context, interval time.Duration) (float64, error) {
  131. cpuTimes, err := p.Times()
  132. if err != nil {
  133. return 0, err
  134. }
  135. now := time.Now()
  136. if interval > 0 {
  137. p.lastCPUTimes = cpuTimes
  138. p.lastCPUTime = now
  139. time.Sleep(interval)
  140. cpuTimes, err = p.Times()
  141. now = time.Now()
  142. if err != nil {
  143. return 0, err
  144. }
  145. } else {
  146. if p.lastCPUTimes == nil {
  147. // invoked first time
  148. p.lastCPUTimes = cpuTimes
  149. p.lastCPUTime = now
  150. return 0, nil
  151. }
  152. }
  153. numcpu := runtime.NumCPU()
  154. delta := (now.Sub(p.lastCPUTime).Seconds()) * float64(numcpu)
  155. ret := calculatePercent(p.lastCPUTimes, cpuTimes, delta, numcpu)
  156. p.lastCPUTimes = cpuTimes
  157. p.lastCPUTime = now
  158. return ret, nil
  159. }
  160. func calculatePercent(t1, t2 *cpu.TimesStat, delta float64, numcpu int) float64 {
  161. if delta == 0 {
  162. return 0
  163. }
  164. delta_proc := t2.Total() - t1.Total()
  165. overall_percent := ((delta_proc / delta) * 100) * float64(numcpu)
  166. return overall_percent
  167. }
  168. // MemoryPercent returns how many percent of the total RAM this process uses
  169. func (p *Process) MemoryPercent() (float32, error) {
  170. return p.MemoryPercentWithContext(context.Background())
  171. }
  172. func (p *Process) MemoryPercentWithContext(ctx context.Context) (float32, error) {
  173. machineMemory, err := mem.VirtualMemory()
  174. if err != nil {
  175. return 0, err
  176. }
  177. total := machineMemory.Total
  178. processMemory, err := p.MemoryInfo()
  179. if err != nil {
  180. return 0, err
  181. }
  182. used := processMemory.RSS
  183. return (100 * float32(used) / float32(total)), nil
  184. }
  185. // CPU_Percent returns how many percent of the CPU time this process uses
  186. func (p *Process) CPUPercent() (float64, error) {
  187. return p.CPUPercentWithContext(context.Background())
  188. }
  189. func (p *Process) CPUPercentWithContext(ctx context.Context) (float64, error) {
  190. crt_time, err := p.CreateTime()
  191. if err != nil {
  192. return 0, err
  193. }
  194. cput, err := p.Times()
  195. if err != nil {
  196. return 0, err
  197. }
  198. created := time.Unix(0, crt_time*int64(time.Millisecond))
  199. totalTime := time.Since(created).Seconds()
  200. if totalTime <= 0 {
  201. return 0, nil
  202. }
  203. return 100 * cput.Total() / totalTime, nil
  204. }