加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
main.go 4.13 KB
一键复制 编辑 原始数据 按行查看 历史
Georg Gentzen 提交于 2020-12-29 22:45 . File position (#67)
// go-callvis: a tool to help visualize the call graph of a Go program.
//
package main
import (
"flag"
"fmt"
"go/build"
"io/ioutil"
"log"
"net"
"net/http"
"net/url"
"os"
"time"
"github.com/pkg/browser"
"golang.org/x/tools/go/buildutil"
)
const Usage = `go-callvis: visualize call graph of a Go program.
Usage:
go-callvis [flags] package
Package should be main package, otherwise -tests flag must be used.
Flags:
`
var (
focusFlag = flag.String("focus", "main", "Focus specific package using name or import path.")
groupFlag = flag.String("group", "pkg", "Grouping functions by packages and/or types [pkg, type] (separated by comma)")
limitFlag = flag.String("limit", "", "Limit package paths to given prefixes (separated by comma)")
ignoreFlag = flag.String("ignore", "", "Ignore package paths containing given prefixes (separated by comma)")
includeFlag = flag.String("include", "", "Include package paths with given prefixes (separated by comma)")
nostdFlag = flag.Bool("nostd", false, "Omit calls to/from packages in standard library.")
nointerFlag = flag.Bool("nointer", false, "Omit calls to unexported functions.")
testFlag = flag.Bool("tests", false, "Include test code.")
graphvizFlag = flag.Bool("graphviz", false, "Use Graphviz's dot program to render images.")
httpFlag = flag.String("http", ":7878", "HTTP service address.")
skipBrowser = flag.Bool("skipbrowser", false, "Skip opening browser.")
outputFile = flag.String("file", "", "output filename - omit to use server mode")
outputFormat = flag.String("format", "svg", "output file format [svg | png | jpg | ...]")
cacheDir = flag.String("cacheDir", "", "Enable caching to avoid unnecessary re-rendering, you can force rendering by adding 'refresh=true' to the URL query or emptying the cache directory")
debugFlag = flag.Bool("debug", false, "Enable verbose log.")
versionFlag = flag.Bool("version", false, "Show version and exit.")
)
func init() {
flag.Var((*buildutil.TagsFlag)(&build.Default.BuildTags), "tags", buildutil.TagsFlagDoc)
// Graphviz options
flag.UintVar(&minlen, "minlen", 2, "Minimum edge length (for wider output).")
flag.Float64Var(&nodesep, "nodesep", 0.35, "Minimum space between two adjacent nodes in the same rank (for taller output).")
}
func logf(f string, a ...interface{}) {
if *debugFlag {
log.Printf(f, a...)
}
}
func parseHTTPAddr(addr string) string {
host, port, _ := net.SplitHostPort(addr)
if host == "" {
host = "localhost"
}
if port == "" {
port = "80"
}
u := url.URL{
Scheme: "http",
Host: fmt.Sprintf("%s:%s", host, port),
}
return u.String()
}
func openBrowser(url string) {
time.Sleep(time.Millisecond * 100)
if err := browser.OpenURL(url); err != nil {
log.Printf("OpenURL error: %v", err)
}
}
func outputDot(fname string, outputFormat string) {
// get cmdline default for analysis
Analysis.OptsSetup()
if e := Analysis.ProcessListArgs(); e != nil {
log.Fatalf("%v\n", e)
}
output, err := Analysis.Render()
if err != nil {
log.Fatalf("%v\n", err)
}
log.Println("writing dot output..")
writeErr := ioutil.WriteFile(fmt.Sprintf("%s.gv", fname), output, 0755)
if writeErr != nil {
log.Fatalf("%v\n", writeErr)
}
log.Printf("converting dot to %s..\n", outputFormat)
_, err = dotToImage(fname, outputFormat, output)
if err != nil {
log.Fatalf("%v\n", err)
}
}
//noinspection GoUnhandledErrorResult
func main() {
flag.Parse()
if *versionFlag {
fmt.Fprintln(os.Stderr, Version())
os.Exit(0)
}
if *debugFlag {
log.SetFlags(log.Lmicroseconds)
}
if flag.NArg() != 1 {
fmt.Fprintf(os.Stderr, Usage)
flag.PrintDefaults()
os.Exit(2)
}
args := flag.Args()
tests := *testFlag
httpAddr := *httpFlag
urlAddr := parseHTTPAddr(httpAddr)
Analysis = new(analysis)
if err := Analysis.DoAnalysis("", tests, args); err != nil {
log.Fatal(err)
}
http.HandleFunc("/", handler)
if *outputFile == "" {
*outputFile = "output"
if !*skipBrowser {
go openBrowser(urlAddr)
}
log.Printf("http serving at %s", urlAddr)
if err := http.ListenAndServe(httpAddr, nil); err != nil {
log.Fatal(err)
}
} else {
outputDot(*outputFile, *outputFormat)
}
}
Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化