Documentation ¶
Overview ¶
Package stack analyzes stack dump of Go processes and simplifies it.
It is mostly useful on servers will large number of identical goroutines, making the crash dump harder to read than strictly necessary.
Example (HTML) ¶
Converts a stack trace from os.Stdin into HTML on os.Stdout, discarding everything else.
package main import ( "io" "log" "os" "github.com/maruel/panicparse/v2/stack" ) func main() { s, _, err := stack.ScanSnapshot(os.Stdin, io.Discard, stack.DefaultOpts()) if err != nil && err != io.EOF { log.Fatal(err) } if s != nil { s.Aggregate(stack.AnyValue).ToHTML(os.Stdout, "") } }
Output:
Example (HttpHandlerMiddleware) ¶
Registers a middleware to trap exceptions and report them on http.Handler.
For demonstration purposes, start a web server that panics and call into it.
package main import ( "bytes" "fmt" "io" "log" "net" "net/http" "os" "path/filepath" "runtime/debug" "github.com/maruel/panicparse/v2/stack" ) func main() { // Start the web server. ln, err := net.Listen("tcp", "localhost:0") if err != nil { log.Fatal(err) } mux := http.ServeMux{} mux.Handle("/", wrapPanic(http.HandlerFunc(panickingHandler))) ch := make(chan error) go func() { ch <- http.Serve(ln, &mux) }() // Call the server once to force a stack trace to be printed. resp, err := http.Get("http://" + ln.Addr().String() + "/") if err != nil { log.Fatal(err) } b, err := io.ReadAll(resp.Body) if err != nil { log.Fatal(err) } resp.Body.Close() if v := string(b); v != "Done" { log.Fatal(v) } // Close the server. if err := ln.Close(); err != nil { log.Fatal(err) } <-ch } // panickingHandler is an http.HandlerFunc that panics. func panickingHandler(w http.ResponseWriter, r *http.Request) { w.Write([]byte("Done")) panic("It happens") } // wrapPanic is a http.Handler middleware that traps panics and print it out to // os.Stdout. func wrapPanic(h http.Handler) http.Handler { return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { defer func() { if v := recover(); v != nil { rawStack := append(debug.Stack(), '\n', '\n') st, _, err := stack.ScanSnapshot(bytes.NewReader(rawStack), io.Discard, stack.DefaultOpts()) if err != nil || len(st.Goroutines) != 1 { fmt.Fprintf(os.Stdout, "recovered: %q\nStack processing failed: %v\nRaw stack:\n%s", v, err, rawStack) return } srcLen := 0 pkgLen := 0 for _, line := range st.Goroutines[0].Stack.Calls { if l := len(fmt.Sprintf("%s:%d", line.SrcName, line.Line)); l > srcLen { srcLen = l } if l := len(filepath.Base(line.Func.ImportPath)); l > pkgLen { pkgLen = l } } buf := bytes.Buffer{} buf.Grow(len(st.Goroutines[0].Stack.Calls) * (40 + srcLen + pkgLen)) for _, line := range st.Goroutines[0].Stack.Calls { if line.Location == stack.Stdlib { continue } args := "<args>" fmt.Fprintf( &buf, " %-*s %-*s %s(%s)\n", pkgLen, line.Func.DirName, srcLen, fmt.Sprintf("%s:%d", line.SrcName, line.Line), line.Func.Name, args) } if st.Goroutines[0].Stack.Elided { io.WriteString(&buf, " (...)\n") } fmt.Fprintf(os.Stdout, "recovered: %q\nParsed stack:\n%s", v, buf.String()) } }() h.ServeHTTP(w, r) }) }
Output: recovered: "It happens" Parsed stack: stack_test example_test.go:243 wrapPanic.func1.1(<args>) stack_test example_test.go:233 panickingHandler(<args>) stack_test example_test.go:293 wrapPanic.func1(<args>)
Example (Simple) ¶
A sample parseStack function expects a stdlib stacktrace from runtime.Stack or debug.Stack and returns the parsed stack object.
package main import ( "bytes" "errors" "fmt" "io" "runtime/debug" "github.com/maruel/panicparse/v2/stack" ) func main() { parseStack := func(rawStack []byte) stack.Stack { s, _, err := stack.ScanSnapshot(bytes.NewReader(rawStack), io.Discard, stack.DefaultOpts()) if err != nil && err != io.EOF { panic(err) } if len(s.Goroutines) > 1 { panic(errors.New("provided stacktrace had more than one goroutine")) } return s.Goroutines[0].Signature.Stack } parsedStack := parseStack(debug.Stack()) fmt.Printf("parsedStack: %#v", parsedStack) }
Output:
Example (Stream) ¶
Process multiple consecutive goroutine snapshots.
package main import ( "bytes" "io" "log" "github.com/maruel/panicparse/v2/stack" ) func main() { // Stream of stack traces: var r io.Reader var w io.Writer opts := stack.DefaultOpts() for { s, suffix, err := stack.ScanSnapshot(r, w, opts) if s != nil { // Process the snapshot... } if err != nil && err != io.EOF { if len(suffix) != 0 { w.Write(suffix) } log.Fatal(err) } // Prepend the suffix that was read to the rest of the input stream to // catch the next snapshot signature: r = io.MultiReader(bytes.NewReader(suffix), r) } }
Output:
Example (Text) ¶
Runs a crashing program and converts it to a dense text format like pp does.
package main import ( "bytes" "fmt" "io" "log" "os" "os/exec" "path/filepath" "github.com/maruel/panicparse/v2/stack" ) func main() { source := `package main import "time" func main() { c := crashy{} go func() { c.die(42.) }() select {} } type crashy struct{} func (c crashy) die(f float32) { time.Sleep(time.Millisecond) panic(int(f)) }` // Skipped error handling to make the example shorter. root, _ := os.MkdirTemp("", "stack") defer os.RemoveAll(root) p := filepath.Join(root, "main.go") os.WriteFile(p, []byte(source), 0600) // Disable both optimization (-N) and inlining (-l). c := exec.Command("go", "run", "-gcflags", "-N -l", p) // This is important, otherwise only the panicking goroutine will be printed. c.Env = append(os.Environ(), "GOTRACEBACK=1") raw, _ := c.CombinedOutput() stream := bytes.NewReader(raw) s, suffix, err := stack.ScanSnapshot(stream, os.Stdout, stack.DefaultOpts()) if err != nil && err != io.EOF { log.Fatal(err) } // Find out similar goroutine traces and group them into buckets. buckets := s.Aggregate(stack.AnyValue).Buckets // Calculate alignment. srcLen := 0 pkgLen := 0 for _, bucket := range buckets { for _, line := range bucket.Signature.Stack.Calls { if l := len(fmt.Sprintf("%s:%d", line.SrcName, line.Line)); l > srcLen { srcLen = l } if l := len(filepath.Base(line.Func.ImportPath)); l > pkgLen { pkgLen = l } } } for _, bucket := range buckets { // Print the goroutine header. extra := "" if s := bucket.SleepString(); s != "" { extra += " [" + s + "]" } if bucket.Locked { extra += " [locked]" } if len(bucket.CreatedBy.Calls) != 0 { extra += fmt.Sprintf(" [Created by %s.%s @ %s:%d]", bucket.CreatedBy.Calls[0].Func.DirName, bucket.CreatedBy.Calls[0].Func.Name, bucket.CreatedBy.Calls[0].SrcName, bucket.CreatedBy.Calls[0].Line) } fmt.Printf("%d: %s%s\n", len(bucket.IDs), bucket.State, extra) // Print the stack lines. for _, line := range bucket.Stack.Calls { fmt.Printf( " %-*s %-*s %s(%s)\n", pkgLen, line.Func.DirName, srcLen, fmt.Sprintf("%s:%d", line.SrcName, line.Line), line.Func.Name, &line.Args) } if bucket.Stack.Elided { io.WriteString(os.Stdout, " (...)\n") } } // If there was any remaining data in the pipe, dump it now. if len(suffix) != 0 { os.Stdout.Write(suffix) } if err == nil { io.Copy(os.Stdout, stream) } }
Output: panic: 42 1: running [Created by main.main @ main.go:7] main main.go:17 crashy.die(42) main main.go:8 main.func1() 1: select (no cases) main main.go:10 main() exit status 2
Index ¶
Examples ¶
Constants ¶
This section is empty.
Variables ¶
This section is empty.
Functions ¶
This section is empty.
Types ¶
type Aggregated ¶
type Aggregated struct { // Snapshot is a pointer to the structure that was used to generate these // buckets. *Snapshot Buckets []*Bucket // contains filtered or unexported fields }
Aggregated is a list of Bucket sorted by repetition count.
type Arg ¶
type Arg struct { // IsAggregate is true if the argument is an aggregate type. If true, the // argument does not contain a value itself, but contains a set of nested // argument fields. If false, the argument contains a single scalar value. IsAggregate bool // Name is a pseudo name given to the argument. Name string // Value is the raw value as found in the stack trace Value uint64 // IsPtr is true if we guess it's a pointer. It's only a guess, it can be // easily confused by a bitmask. IsPtr bool // IsOffsetTooLarge is true if the argument's frame offset was too large, // preventing the argument from being printed in the stack trace. IsOffsetTooLarge bool // IsInaccurate determines if Value is inaccurate. Stacks could have inaccurate values // for arguments passed in registers. Go 1.18 prints a ? for these values. IsInaccurate bool // Fields are the fields/elements of aggregate-typed arguments. Fields Args // contains filtered or unexported fields }
Arg is an argument on a Call.
type Args ¶
type Args struct { // Values is the arguments as shown on the stack trace. They are mangled via // simplification. Values []Arg // Processed is the arguments generated from processing the source files. It // can have a length lower than Values. Processed []string // Elided when set means there was a trailing ", ...". Elided bool // contains filtered or unexported fields }
Args is a series of function call arguments.
type Bucket ¶
type Bucket struct { // Signature is the generalized signature for this bucket. Signature // IDs is the ID of each Goroutine with this Signature. IDs []int // First is true if this Bucket contains the first goroutine, e.g. the one // Signature that likely generated the panic() call, if any. First bool // contains filtered or unexported fields }
Bucket is a stack trace signature and the list of goroutines that fits this signature.
type Call ¶
type Call struct { // Func is the fully qualified function name (encoded). Func Func // Args is the call arguments. Args Args // RemoteSrcPath is the full path name of the source file as seen in the // trace. RemoteSrcPath string // Line is the line number. Line int // SrcName is the base file name of the source file. SrcName string // DirSrc is one directory plus the file name of the source file. It is a // subset of RemoteSrcPath. DirSrc string // LocalSrcPath is the full path name of the source file as seen in the host, // if found. LocalSrcPath string // RelSrcPath is the relative path to GOROOT, GOPATH or LocalGoMods. RelSrcPath string // ImportPath is the fully qualified import path as found on disk (when // Opts.GuessPaths was set). Defaults to Func.ImportPath otherwise. // // In the case of package "main", it returns the underlying path to the main // package instead of "main" if Opts.GuessPaths was set. ImportPath string // Location is the source location, if determined. Location Location // contains filtered or unexported fields }
Call is an item in the stack trace.
All paths in this struct are in POSIX format, using "/" as path separator.
type Func ¶
type Func struct { // Complete is the complete reference. It can be ambiguous in case where a // path contains dots. Complete string // ImportPath is the directory name for this function reference, or "main" if // it was in package main. The package name may not match. ImportPath string // DirName is the directory name containing the package in which the function // is. Normally this matches the package name, but sometimes there's smartass // folks that use a different directory name than the package name. DirName string // Name is the function name or fully quality method name. Name string // IsExported is true if the function is exported. IsExported bool // IsPkgMain is true if it is in the main package. IsPkgMain bool // contains filtered or unexported fields }
Func is a function call in a goroutine stack trace.
func (*Func) Init ¶
Init parses the raw function call line from a goroutine stack trace.
Go stack traces print a mangled function call, this wrapper unmangle the string before printing and adds other filtering methods.
The main caveat is that for calls in package main, the package import URL is left out.
type Goroutine ¶
type Goroutine struct { // Signature is the stack trace, internal bits, state, which call site // created it, etc. Signature // ID is the goroutine id. ID int // First is the goroutine first printed, normally the one that crashed. First bool // RaceWrite is true if a race condition was detected, and this goroutine was // race on a write operation, otherwise it was a read. RaceWrite bool // RaceAddr is set to the address when a data race condition was detected. // Otherwise it is 0. RaceAddr uint64 // contains filtered or unexported fields }
Goroutine represents the state of one goroutine, including the stack trace.
type Location ¶
type Location int
Location is the source location, if determined.
const ( // LocationUnknown is the default value when Opts.GuessPaths was false. LocationUnknown Location = iota // GoMod is a go module, it is outside $GOPATH and is inside a directory // containing a go.mod file. This is considered a local copy. GoMod // GOPATH is in $GOPATH/src. This is either a dependency fetched via // GO111MODULE=off or intentionally fetched this way. There is no guaranteed // that the local copy is pristine. GOPATH // GoPkg is in $GOPATH/pkg/mod. This is a dependency fetched via go module. // It is considered to be an unmodified external dependency. GoPkg // Stdlib is when it is a Go standard library function. This includes the 'go // test' generated main executable. Stdlib )
type Opts ¶
type Opts struct { // LocalGOROOT is GOROOT with "/" as path separator. No trailing "/". Can be // unset. LocalGOROOT string // LocalGOPATHs is GOPATH with "/" as path separator. No trailing "/". Can be // unset. LocalGOPATHs []string // NameArguments tells panicparse to find the recurring pointer values and // give them pseudo 'names'. // // Since the algorithm is O(n²), this can be worth disabling on live servers. NameArguments bool // GuessPaths tells panicparse to guess local RemoteGOROOT and GOPATH for // what was found in the snapshot. // // Initializes in Snapshot the following members: RemoteGOROOT, // RemoteGOPATHs, LocalGomoduleRoot and GomodImportPath. // // This is done by scanning the local disk, so be warned of performance // impact. GuessPaths bool // AnalyzeSources tells panicparse to processes source files to improve calls // to be more descriptive. // // Requires GuessPaths to be true. AnalyzeSources bool // contains filtered or unexported fields }
Opts represents options to process the snapshot.
func DefaultOpts ¶
func DefaultOpts() *Opts
DefaultOpts returns default options to process the snapshot.
type Signature ¶
type Signature struct { // State is the goroutine state at the time of the snapshot. // // Use git grep 'gopark(|unlock)\(' to find them all plus everything listed // in runtime/traceback.go. Valid values includes: // - chan send, chan receive, select // - finalizer wait, mark wait (idle), // - Concurrent GC wait, GC sweep wait, force gc (idle) // - IO wait, panicwait // - semacquire, semarelease // - sleep, timer goroutine (idle) // - trace reader (blocked) // Stuck cases: // - chan send (nil chan), chan receive (nil chan), select (no cases) // Runnable states: // - idle, runnable, running, syscall, waiting, dead, enqueue, copystack, // Scan states: // - scan, scanrunnable, scanrunning, scansyscall, scanwaiting, scandead, // scanenqueue // // When running under the race detector, the values are 'running' or // 'finished'. State string // CreatedBy is the call stack that created this goroutine, if applicable. // // Normally, the stack is a single Call. // // When the race detector is enabled, a full stack snapshot is available. CreatedBy Stack // SleepMin is the wait time in minutes, if applicable. // // Not set when running under the race detector. SleepMin int // SleepMax is the wait time in minutes, if applicable. // // Not set when running under the race detector. SleepMax int // Stack is the call stack. Stack Stack // Locked is set if the goroutine was locked to an OS thread. // // Not set when running under the race detector. Locked bool // contains filtered or unexported fields }
Signature represents the signature of one or multiple goroutines.
It is effectively the stack trace plus the goroutine internal bits, like it's state, if it is thread locked, which call site created this goroutine, etc.
func (*Signature) SleepString ¶
SleepString returns a string "N-M minutes" if the goroutine(s) slept for a long time.
Returns an empty string otherwise.
type Similarity ¶
type Similarity int
Similarity is the level at which two call lines arguments must match to be considered similar enough to coalesce them.
const ( // ExactFlags requires same bits (e.g. Locked). ExactFlags Similarity = iota // ExactLines requests the exact same arguments on the call line. ExactLines // AnyPointer considers different pointers a similar call line. AnyPointer // AnyValue accepts any value as similar call line. AnyValue )
type Snapshot ¶
type Snapshot struct { // Goroutines is the Goroutines found. // // They are in the order that they were printed. Goroutines []*Goroutine // LocalGOROOT is copied from Opts. LocalGOROOT string // LocalGOPATHs is copied from Opts. LocalGOPATHs []string // RemoteGOROOT is the GOROOT as detected in the traceback, not the on the // host. // // It can be empty if no root was determined, for example the traceback // contains only non-stdlib source references. RemoteGOROOT string // RemoteGOPATHs is the GOPATH as detected in the traceback, with the value // being the corresponding path mapped to the host if found. // // It can be empty if only stdlib code is in the traceback or if no local // sources were matched up. In the general case there is only one entry in // the map. RemoteGOPATHs map[string]string // LocalGomods are the root directories containing go.mod or that directly // contained source code as detected in the traceback, with the value being // the corresponding import path found in the go.mod file. // // Uses "/" as path separator. No trailing "/". // // Because of the "replace" statement in go.mod, there can be multiple root // directories. A file run by "go run" is also considered a go module to (a // certain extent). // // It is initialized by findRoots(). // // Unlike GOROOT and GOPATH, it only works with stack traces created in the // local file system, hence "Local" prefix. LocalGomods map[string]string // contains filtered or unexported fields }
Snapshot is a parsed runtime.Stack() or race detector dump.
func ScanSnapshot ¶
ScanSnapshot scans the Reader for the output from runtime.Stack() in br.
Returns nil *Snapshot if no stack trace was detected.
If a Snapshot is returned, you can call the function again to find another trace, or do io.Copy(br, out) to flush the rest of the stream.
ParseSnapshot processes the output from runtime.Stack() or the race detector.
Returns a nil *Snapshot if no stack trace was detected and SearchSnapshot() was a false positive.
Returns io.EOF if all of reader was read.
The suffix of the stack trace is returned as []byte.
It pipes anything not detected as a panic stack trace from r into out. It assumes there is junk before the actual stack trace. The junk is streamed to out.
func (*Snapshot) Aggregate ¶
func (s *Snapshot) Aggregate(similar Similarity) *Aggregated
Aggregate merges similar goroutines into buckets.
The buckets are ordered in library provided order of relevancy. You can reorder at your choosing.
type Stack ¶
type Stack struct { // Calls is the call stack. First is original function, last is leaf // function. Calls []Call // Elided is set when there's >100 items in Stack, currently hardcoded in // package runtime. Elided bool // contains filtered or unexported fields }
Stack is a call stack.