package cmd import ( "net/http" // FIXME: importing this package has the side-effect of poisoning the http.DefaultServeMux // FIXME: with the /debug/pprof endpoints "context" "net" "net/http/pprof" ) type PProfServer struct { cc chan PprofServerControlMsg state PprofServerControlMsg listener net.Listener } type PprofServerControlMsg struct { // Whether the server should listen for requests on the given address Run bool // Must be set if Run is true, undefined otherwise HttpListenAddress string } func NewPProfServer(ctx context.Context) *PProfServer { s := &PProfServer{ cc: make(chan PprofServerControlMsg), } go s.controlLoop(ctx) return s } func (s *PProfServer) controlLoop(ctx context.Context) { outer: for { var msg PprofServerControlMsg select { case <-ctx.Done(): if s.listener != nil { s.listener.Close() } break outer case msg = <-s.cc: // proceed } var err error if msg.Run && s.listener == nil { s.listener, err = net.Listen("tcp", msg.HttpListenAddress) if err != nil { s.listener = nil continue } // FIXME: because net/http/pprof does not provide a mux, mux := http.NewServeMux() mux.Handle("/debug/pprof/", http.HandlerFunc(pprof.Index)) mux.Handle("/debug/pprof/cmdline", http.HandlerFunc(pprof.Cmdline)) mux.Handle("/debug/pprof/profile", http.HandlerFunc(pprof.Profile)) mux.Handle("/debug/pprof/symbol", http.HandlerFunc(pprof.Symbol)) mux.Handle("/debug/pprof/trace", http.HandlerFunc(pprof.Trace)) go http.Serve(s.listener, mux) continue } if !msg.Run && s.listener != nil { s.listener.Close() s.listener = nil continue } } } func (s *PProfServer) Control(msg PprofServerControlMsg) { s.cc <- msg }