process_posix.go 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180
  1. // +build linux freebsd openbsd darwin solaris
  2. package process
  3. import (
  4. "context"
  5. "fmt"
  6. "os"
  7. "os/user"
  8. "path/filepath"
  9. "strconv"
  10. "strings"
  11. "syscall"
  12. "github.com/shirou/gopsutil/internal/common"
  13. "golang.org/x/sys/unix"
  14. )
  15. // POSIX
  16. func getTerminalMap() (map[uint64]string, error) {
  17. ret := make(map[uint64]string)
  18. var termfiles []string
  19. d, err := os.Open("/dev")
  20. if err != nil {
  21. return nil, err
  22. }
  23. defer d.Close()
  24. devnames, err := d.Readdirnames(-1)
  25. if err != nil {
  26. return nil, err
  27. }
  28. for _, devname := range devnames {
  29. if strings.HasPrefix(devname, "/dev/tty") {
  30. termfiles = append(termfiles, "/dev/tty/"+devname)
  31. }
  32. }
  33. var ptsnames []string
  34. ptsd, err := os.Open("/dev/pts")
  35. if err != nil {
  36. ptsnames, _ = filepath.Glob("/dev/ttyp*")
  37. if ptsnames == nil {
  38. return nil, err
  39. }
  40. }
  41. defer ptsd.Close()
  42. if ptsnames == nil {
  43. defer ptsd.Close()
  44. ptsnames, err = ptsd.Readdirnames(-1)
  45. if err != nil {
  46. return nil, err
  47. }
  48. for _, ptsname := range ptsnames {
  49. termfiles = append(termfiles, "/dev/pts/"+ptsname)
  50. }
  51. } else {
  52. termfiles = ptsnames
  53. }
  54. for _, name := range termfiles {
  55. stat := unix.Stat_t{}
  56. if err = unix.Stat(name, &stat); err != nil {
  57. return nil, err
  58. }
  59. rdev := uint64(stat.Rdev)
  60. ret[rdev] = strings.Replace(name, "/dev", "", -1)
  61. }
  62. return ret, nil
  63. }
  64. // isMount is a port of python's os.path.ismount()
  65. // https://github.com/python/cpython/blob/08ff4369afca84587b1c82034af4e9f64caddbf2/Lib/posixpath.py#L186-L216
  66. // https://docs.python.org/3/library/os.path.html#os.path.ismount
  67. func isMount(path string) bool {
  68. // Check symlinkness with os.Lstat; unix.DT_LNK is not portable
  69. fileInfo, err := os.Lstat(path)
  70. if err != nil {
  71. return false
  72. }
  73. if fileInfo.Mode() & os.ModeSymlink != 0 {
  74. return false
  75. }
  76. var stat1 unix.Stat_t
  77. if err := unix.Lstat(path, &stat1); err != nil {
  78. return false
  79. }
  80. parent := filepath.Join(path, "..")
  81. var stat2 unix.Stat_t
  82. if err := unix.Lstat(parent, &stat2); err != nil {
  83. return false
  84. }
  85. return stat1.Dev != stat2.Dev || stat1.Ino == stat2.Ino
  86. }
  87. func PidExistsWithContext(ctx context.Context, pid int32) (bool, error) {
  88. if pid <= 0 {
  89. return false, fmt.Errorf("invalid pid %v", pid)
  90. }
  91. proc, err := os.FindProcess(int(pid))
  92. if err != nil {
  93. return false, err
  94. }
  95. if isMount(common.HostProc()) { // if /<HOST_PROC>/proc exists and is mounted, check if /<HOST_PROC>/proc/<PID> folder exists
  96. _, err := os.Stat(common.HostProc(strconv.Itoa(int(pid))))
  97. if os.IsNotExist(err) {
  98. return false, nil
  99. }
  100. return err == nil, err
  101. }
  102. // procfs does not exist or is not mounted, check PID existence by signalling the pid
  103. err = proc.Signal(syscall.Signal(0))
  104. if err == nil {
  105. return true, nil
  106. }
  107. if err.Error() == "os: process already finished" {
  108. return false, nil
  109. }
  110. errno, ok := err.(syscall.Errno)
  111. if !ok {
  112. return false, err
  113. }
  114. switch errno {
  115. case syscall.ESRCH:
  116. return false, nil
  117. case syscall.EPERM:
  118. return true, nil
  119. }
  120. return false, err
  121. }
  122. func (p *Process) SendSignalWithContext(ctx context.Context, sig syscall.Signal) error {
  123. process, err := os.FindProcess(int(p.Pid))
  124. if err != nil {
  125. return err
  126. }
  127. err = process.Signal(sig)
  128. if err != nil {
  129. return err
  130. }
  131. return nil
  132. }
  133. func (p *Process) SuspendWithContext(ctx context.Context) error {
  134. return p.SendSignalWithContext(ctx, unix.SIGSTOP)
  135. }
  136. func (p *Process) ResumeWithContext(ctx context.Context) error {
  137. return p.SendSignalWithContext(ctx, unix.SIGCONT)
  138. }
  139. func (p *Process) TerminateWithContext(ctx context.Context) error {
  140. return p.SendSignalWithContext(ctx, unix.SIGTERM)
  141. }
  142. func (p *Process) KillWithContext(ctx context.Context) error {
  143. return p.SendSignalWithContext(ctx, unix.SIGKILL)
  144. }
  145. func (p *Process) UsernameWithContext(ctx context.Context) (string, error) {
  146. uids, err := p.UidsWithContext(ctx)
  147. if err != nil {
  148. return "", err
  149. }
  150. if len(uids) > 0 {
  151. u, err := user.LookupId(strconv.Itoa(int(uids[0])))
  152. if err != nil {
  153. return "", err
  154. }
  155. return u.Username, nil
  156. }
  157. return "", nil
  158. }