1
0
mirror of https://github.com/google/gops.git synced 2024-11-24 08:22:25 +02:00
gops/gops.go

151 lines
2.7 KiB
Go
Raw Normal View History

2016-11-04 01:56:19 +02:00
// Copyright 2013 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.
// Program gops is a tool to list currently running Go processes.
2016-11-03 23:01:55 +02:00
package main
import (
2016-11-04 07:32:46 +02:00
"flag"
2016-11-03 23:01:55 +02:00
"fmt"
2016-11-04 07:32:46 +02:00
"io/ioutil"
2016-11-03 23:01:55 +02:00
"log"
2016-11-04 07:32:46 +02:00
"net"
"os"
2016-11-03 23:01:55 +02:00
2016-11-04 07:32:46 +02:00
"hello/gops/agent/signal"
2016-11-03 23:01:55 +02:00
"hello/gops/internal/objfile"
ps "github.com/keybase/go-ps"
)
2016-11-04 07:32:46 +02:00
const helpText = `Usage: gops is a tool to list and diagnose Go processes.
gops Lists all Go processes currently running.
Advanced tools, requires gops agent:
gops stack -p pid Prints the stack trace of the process.
gops gc -p pid Runs the garbage collector.
gops gcstats -p pid Prints the GC stats of the process.
gops help Prints this help message.
`
var (
pid = flag.Int("p", -1, "")
stack = flag.Bool("stack", false, "")
gc = flag.Bool("gc", false, "")
gcstats = flag.Bool("gcstats", false, "")
help = flag.Bool("help", false, "")
)
2016-11-03 23:01:55 +02:00
func main() {
2016-11-04 07:32:46 +02:00
flag.Usage = usage
flag.Parse()
if len(os.Args) < 2 {
list()
return
}
if *pid == -1 {
usage()
}
if *help {
usage()
}
if *stack {
out, err := cmd(signal.Stack)
exit(err)
fmt.Println(out)
}
if *gc {
_, err := cmd(signal.GC)
exit(err)
}
if *gcstats {
out, err := cmd(signal.GC)
exit(err)
fmt.Println(out)
}
}
func cmd(c byte) (string, error) {
sock := fmt.Sprintf("/tmp/gops%d.sock", *pid)
conn, err := net.Dial("unix", sock)
if err != nil {
return "", err
}
if _, err := conn.Write([]byte{c}); err != nil {
return "", err
}
all, err := ioutil.ReadAll(conn)
if err != nil {
return "", err
}
return string(all), nil
}
func list() {
2016-11-03 23:01:55 +02:00
pss, err := ps.Processes()
if err != nil {
log.Fatal(err)
}
var undetermined int
for _, pr := range pss {
name, err := pr.Path()
if err != nil {
undetermined++
continue
}
ok, err := isGo(name)
if err != nil {
2016-11-04 04:33:43 +02:00
// TODO(jbd): worth to report the number?
2016-11-03 23:01:55 +02:00
continue
}
if ok {
2016-11-04 04:33:43 +02:00
fmt.Printf("%d\t%v\t(%v)\n", pr.Pid(), pr.Executable(), name)
2016-11-03 23:01:55 +02:00
}
}
2016-11-04 01:56:19 +02:00
if undetermined > 0 {
2016-11-04 02:52:29 +02:00
fmt.Printf("\n%d processes left undetermined\n", undetermined)
2016-11-04 01:56:19 +02:00
}
2016-11-03 23:01:55 +02:00
}
func isGo(filename string) (ok bool, err error) {
obj, err := objfile.Open(filename)
if err != nil {
return false, err
}
defer obj.Close()
symbols, err := obj.Symbols()
if err != nil {
return false, err
}
// TODO(jbd): find a faster way to determine Go programs
2016-11-04 01:56:19 +02:00
// looping over the symbols is a joke.
2016-11-03 23:01:55 +02:00
for _, s := range symbols {
if s.Name == "runtime.buildVersion" {
return true, nil
}
}
return false, nil
}
2016-11-04 07:32:46 +02:00
func usage() {
fmt.Fprintf(os.Stderr, "%v\n", helpText)
os.Exit(1)
}
func exit(err error) {
if err != nil {
fmt.Fprintf(os.Stderr, "%v\n", err)
os.Exit(1)
}
}