2018-08-27 19:10:55 +02:00
|
|
|
package daemon
|
|
|
|
|
|
|
|
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"
|
2019-03-22 20:45:27 +01:00
|
|
|
|
2020-05-17 13:56:48 +02:00
|
|
|
"github.com/prometheus/client_golang/prometheus/promhttp"
|
2020-04-11 15:49:41 +02:00
|
|
|
"golang.org/x/net/websocket"
|
|
|
|
|
2019-03-22 20:45:27 +01:00
|
|
|
"github.com/zrepl/zrepl/daemon/job"
|
2020-04-11 15:49:41 +02:00
|
|
|
"github.com/zrepl/zrepl/daemon/logging/trace"
|
2018-08-27 19:10:55 +02:00
|
|
|
)
|
|
|
|
|
2018-08-27 19:13:35 +02:00
|
|
|
type pprofServer struct {
|
2018-08-27 19:10:55 +02:00
|
|
|
cc chan 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
|
|
|
|
}
|
|
|
|
|
2018-08-27 19:13:35 +02:00
|
|
|
func NewPProfServer(ctx context.Context) *pprofServer {
|
2018-08-27 19:10:55 +02:00
|
|
|
|
2018-08-27 19:13:35 +02:00
|
|
|
s := &pprofServer{
|
2018-08-27 19:10:55 +02:00
|
|
|
cc: make(chan PprofServerControlMsg),
|
|
|
|
}
|
|
|
|
|
|
|
|
go s.controlLoop(ctx)
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
2018-08-27 19:13:35 +02:00
|
|
|
func (s *pprofServer) controlLoop(ctx context.Context) {
|
2018-08-27 19:10:55 +02:00
|
|
|
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))
|
2020-05-17 13:56:48 +02:00
|
|
|
mux.Handle("/metrics", promhttp.Handler())
|
2020-04-11 15:49:41 +02:00
|
|
|
mux.Handle("/debug/zrepl/activity-trace", websocket.Handler(trace.ChrometraceClientWebsocketHandler))
|
2019-03-22 20:45:27 +01:00
|
|
|
go func() {
|
|
|
|
err := http.Serve(s.listener, mux)
|
|
|
|
if ctx.Err() != nil {
|
|
|
|
return
|
|
|
|
} else if err != nil {
|
|
|
|
job.GetLogger(ctx).WithError(err).Error("pprof server serve error")
|
|
|
|
}
|
|
|
|
}()
|
2018-08-27 19:10:55 +02:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if !msg.Run && s.listener != nil {
|
|
|
|
s.listener.Close()
|
|
|
|
s.listener = nil
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-27 19:13:35 +02:00
|
|
|
func (s *pprofServer) Control(msg PprofServerControlMsg) {
|
2018-08-27 19:10:55 +02:00
|
|
|
s.cc <- msg
|
|
|
|
}
|