2023-12-08 17:03:46 -05:00
|
|
|
// Copyright 2023 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 debug_test
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"io"
|
|
|
|
|
"log"
|
|
|
|
|
"os"
|
|
|
|
|
"os/exec"
|
|
|
|
|
"runtime/debug"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// ExampleSetCrashOutput_monitor shows an example of using
|
|
|
|
|
// [debug.SetCrashOutput] to direct crashes to a "monitor" process,
|
|
|
|
|
// for automated crash reporting. The monitor is the same executable,
|
|
|
|
|
// invoked in a special mode indicated by an environment variable.
|
|
|
|
|
func ExampleSetCrashOutput_monitor() {
|
|
|
|
|
appmain()
|
|
|
|
|
|
|
|
|
|
// This Example doesn't actually run as a test because its
|
|
|
|
|
// purpose is to crash, so it has no "Output:" comment
|
|
|
|
|
// within the function body.
|
|
|
|
|
//
|
|
|
|
|
// To observe the monitor in action, replace the entire text
|
|
|
|
|
// of this comment with "Output:" and run this command:
|
|
|
|
|
//
|
|
|
|
|
// $ go test -run=ExampleSetCrashOutput_monitor runtime/debug
|
|
|
|
|
// panic: oops
|
|
|
|
|
// ...stack...
|
|
|
|
|
// monitor: saved crash report at /tmp/10804884239807998216.crash
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// appmain represents the 'main' function of your application.
|
|
|
|
|
func appmain() {
|
|
|
|
|
monitor()
|
|
|
|
|
|
|
|
|
|
// Run the application.
|
|
|
|
|
println("hello")
|
|
|
|
|
panic("oops")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// monitor starts the monitor process, which performs automated
|
|
|
|
|
// crash reporting. Call this function immediately within main.
|
|
|
|
|
//
|
|
|
|
|
// This function re-executes the same executable as a child process,
|
|
|
|
|
// in a special mode. In that mode, the call to monitor will never
|
|
|
|
|
// return.
|
|
|
|
|
func monitor() {
|
|
|
|
|
const monitorVar = "RUNTIME_DEBUG_MONITOR"
|
|
|
|
|
if os.Getenv(monitorVar) != "" {
|
|
|
|
|
// This is the monitor (child) process.
|
|
|
|
|
log.SetFlags(0)
|
|
|
|
|
log.SetPrefix("monitor: ")
|
|
|
|
|
|
|
|
|
|
crash, err := io.ReadAll(os.Stdin)
|
|
|
|
|
if err != nil {
|
|
|
|
|
log.Fatalf("failed to read from input pipe: %v", err)
|
|
|
|
|
}
|
|
|
|
|
if len(crash) == 0 {
|
|
|
|
|
// Parent process terminated without reporting a crash.
|
|
|
|
|
os.Exit(0)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Save the crash report securely in the file system.
|
|
|
|
|
f, err := os.CreateTemp("", "*.crash")
|
|
|
|
|
if err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
if _, err := f.Write(crash); err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
if err := f.Close(); err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
log.Fatalf("saved crash report at %s", f.Name())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// This is the application process.
|
|
|
|
|
// Fork+exec the same executable in monitor mode.
|
|
|
|
|
exe, err := os.Executable()
|
|
|
|
|
if err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
cmd := exec.Command(exe, "-test.run=ExampleSetCrashOutput_monitor")
|
|
|
|
|
cmd.Env = append(os.Environ(), monitorVar+"=1")
|
|
|
|
|
cmd.Stderr = os.Stderr
|
|
|
|
|
cmd.Stdout = os.Stderr
|
|
|
|
|
pipe, err := cmd.StdinPipe()
|
|
|
|
|
if err != nil {
|
|
|
|
|
log.Fatalf("StdinPipe: %v", err)
|
|
|
|
|
}
|
2024-05-15 17:41:56 -04:00
|
|
|
debug.SetCrashOutput(pipe.(*os.File), debug.CrashOptions{}) // (this conversion is safe)
|
2023-12-08 17:03:46 -05:00
|
|
|
if err := cmd.Start(); err != nil {
|
|
|
|
|
log.Fatalf("can't start monitor: %v", err)
|
|
|
|
|
}
|
|
|
|
|
// Now return and start the application proper...
|
|
|
|
|
}
|