2017-08-29 04:32:14 +02:00
|
|
|
// Copyright 2017 The Go Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
// Package goprocess reports the Go processes running on a host.
|
|
|
|
package goprocess
|
|
|
|
|
|
|
|
import (
|
|
|
|
"os"
|
|
|
|
"sync"
|
|
|
|
|
2017-09-04 20:53:45 +02:00
|
|
|
goversion "rsc.io/goversion/version"
|
|
|
|
|
2017-08-29 04:32:14 +02:00
|
|
|
"github.com/google/gops/internal"
|
|
|
|
ps "github.com/keybase/go-ps"
|
|
|
|
)
|
|
|
|
|
2017-09-04 20:45:49 +02:00
|
|
|
// P represents a Go process.
|
|
|
|
type P struct {
|
2017-08-29 04:32:14 +02:00
|
|
|
PID int
|
|
|
|
PPID int
|
|
|
|
Exec string
|
|
|
|
Path string
|
|
|
|
BuildVersion string
|
|
|
|
Agent bool
|
|
|
|
}
|
|
|
|
|
2017-09-04 20:45:49 +02:00
|
|
|
// FindAll returns all the Go processes currently running on this host.
|
|
|
|
func FindAll() []P {
|
2020-09-07 23:23:34 +02:00
|
|
|
const concurrencyLimit = 10 // max number of concurrent workers
|
2017-08-29 04:32:14 +02:00
|
|
|
pss, err := ps.Processes()
|
|
|
|
if err != nil {
|
2018-03-11 07:24:15 +02:00
|
|
|
return nil
|
2017-08-29 04:32:14 +02:00
|
|
|
}
|
2020-09-10 10:12:57 +02:00
|
|
|
return findAll(pss, isGo, concurrencyLimit)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Allows to inject isGo for testing.
|
|
|
|
type isGoFunc func(ps.Process) (path, version string, agent, ok bool, err error)
|
2017-08-29 04:32:14 +02:00
|
|
|
|
2020-09-10 10:12:57 +02:00
|
|
|
func findAll(pss []ps.Process, isGo isGoFunc, concurrencyLimit int) []P {
|
2020-09-07 23:23:34 +02:00
|
|
|
input := make(chan ps.Process, len(pss))
|
|
|
|
output := make(chan P, len(pss))
|
|
|
|
|
|
|
|
for _, ps := range pss {
|
|
|
|
input <- ps
|
|
|
|
}
|
|
|
|
close(input)
|
|
|
|
|
2017-08-29 04:32:14 +02:00
|
|
|
var wg sync.WaitGroup
|
2020-09-07 23:23:34 +02:00
|
|
|
wg.Add(concurrencyLimit) // used to wait for workers to be finished
|
2017-08-29 04:32:14 +02:00
|
|
|
|
2020-09-07 23:23:34 +02:00
|
|
|
// Run concurrencyLimit of workers until there
|
|
|
|
// is no more processes to be checked in the input channel.
|
|
|
|
for i := 0; i < concurrencyLimit; i++ {
|
2017-08-29 04:32:14 +02:00
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
|
|
|
|
2020-09-07 23:23:34 +02:00
|
|
|
for pr := range input {
|
|
|
|
path, version, agent, ok, err := isGo(pr)
|
|
|
|
if err != nil {
|
|
|
|
// TODO(jbd): Return a list of errors.
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if !ok {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
output <- P{
|
|
|
|
PID: pr.Pid(),
|
|
|
|
PPID: pr.PPid(),
|
|
|
|
Exec: pr.Executable(),
|
|
|
|
Path: path,
|
|
|
|
BuildVersion: version,
|
|
|
|
Agent: agent,
|
|
|
|
}
|
2018-03-11 07:24:15 +02:00
|
|
|
}
|
2017-08-29 04:32:14 +02:00
|
|
|
}()
|
|
|
|
}
|
2020-09-07 23:23:34 +02:00
|
|
|
wg.Wait() // wait until all workers are finished
|
|
|
|
close(output) // no more results to be waited for
|
|
|
|
|
2018-03-11 07:24:15 +02:00
|
|
|
var results []P
|
2020-09-07 23:23:34 +02:00
|
|
|
for p := range output {
|
2018-03-11 07:24:15 +02:00
|
|
|
results = append(results, p)
|
|
|
|
}
|
2017-08-29 04:32:14 +02:00
|
|
|
return results
|
|
|
|
}
|
|
|
|
|
2017-09-04 20:45:49 +02:00
|
|
|
// Find finds info about the process identified with the given PID.
|
|
|
|
func Find(pid int) (p P, ok bool, err error) {
|
|
|
|
pr, err := ps.FindProcess(pid)
|
|
|
|
if err != nil {
|
|
|
|
return P{}, false, err
|
|
|
|
}
|
|
|
|
path, version, agent, ok, err := isGo(pr)
|
|
|
|
if !ok {
|
|
|
|
return P{}, false, nil
|
|
|
|
}
|
|
|
|
return P{
|
|
|
|
PID: pr.Pid(),
|
|
|
|
PPID: pr.PPid(),
|
|
|
|
Exec: pr.Executable(),
|
|
|
|
Path: path,
|
|
|
|
BuildVersion: version,
|
|
|
|
Agent: agent,
|
|
|
|
}, true, nil
|
|
|
|
}
|
|
|
|
|
2017-08-29 04:32:14 +02:00
|
|
|
// isGo looks up the runtime.buildVersion symbol
|
|
|
|
// in the process' binary and determines if the process
|
|
|
|
// if a Go process or not. If the process is a Go process,
|
|
|
|
// it reports PID, binary name and full path of the binary.
|
|
|
|
func isGo(pr ps.Process) (path, version string, agent, ok bool, err error) {
|
|
|
|
if pr.Pid() == 0 {
|
|
|
|
// ignore system process
|
|
|
|
return
|
|
|
|
}
|
|
|
|
path, err = pr.Path()
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
2017-09-04 20:53:45 +02:00
|
|
|
var versionInfo goversion.Version
|
|
|
|
versionInfo, err = goversion.ReadExe(path)
|
2017-08-29 04:32:14 +02:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
2017-09-04 20:53:45 +02:00
|
|
|
ok = true
|
|
|
|
version = versionInfo.Release
|
2017-08-29 04:32:14 +02:00
|
|
|
pidfile, err := internal.PIDFile(pr.Pid())
|
|
|
|
if err == nil {
|
|
|
|
_, err := os.Stat(pidfile)
|
|
|
|
agent = err == nil
|
|
|
|
}
|
|
|
|
return path, version, agent, ok, nil
|
|
|
|
}
|