2017-09-17 23:54:23 +02:00
|
|
|
package cmd
|
|
|
|
|
|
|
|
import (
|
2017-11-18 20:34:28 +01:00
|
|
|
"bytes"
|
2017-09-17 23:54:23 +02:00
|
|
|
"context"
|
2017-11-18 20:34:28 +01:00
|
|
|
"encoding/json"
|
2017-09-17 23:54:23 +02:00
|
|
|
"fmt"
|
|
|
|
"github.com/spf13/cobra"
|
|
|
|
"io"
|
|
|
|
golog "log"
|
|
|
|
"net"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"os"
|
|
|
|
)
|
|
|
|
|
|
|
|
var controlCmd = &cobra.Command{
|
|
|
|
Use: "control",
|
|
|
|
Short: "control zrepl daemon",
|
|
|
|
}
|
|
|
|
|
|
|
|
var pprofCmd = &cobra.Command{
|
|
|
|
Use: "pprof cpu OUTFILE",
|
|
|
|
Short: "pprof CPU of daemon to OUTFILE (- for stdout)",
|
|
|
|
Run: doControlPProf,
|
|
|
|
}
|
|
|
|
var pprofCmdArgs struct {
|
|
|
|
seconds int64
|
|
|
|
}
|
|
|
|
|
2017-11-18 20:34:28 +01:00
|
|
|
var controlVersionCmd = &cobra.Command{
|
|
|
|
Use: "version",
|
|
|
|
Short: "print version of running zrepl daemon",
|
|
|
|
Run: doControLVersionCmd,
|
|
|
|
}
|
|
|
|
|
2017-09-17 23:54:23 +02:00
|
|
|
func init() {
|
|
|
|
RootCmd.AddCommand(controlCmd)
|
|
|
|
controlCmd.AddCommand(pprofCmd)
|
|
|
|
pprofCmd.Flags().Int64Var(&pprofCmdArgs.seconds, "seconds", 30, "seconds to profile")
|
2017-11-18 20:34:28 +01:00
|
|
|
controlCmd.AddCommand(controlVersionCmd)
|
|
|
|
}
|
|
|
|
|
|
|
|
func controlHttpClient() (client http.Client, err error) {
|
|
|
|
|
|
|
|
conf, err := ParseConfig(rootArgs.configFile)
|
|
|
|
if err != nil {
|
|
|
|
return http.Client{}, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return http.Client{
|
|
|
|
Transport: &http.Transport{
|
|
|
|
DialContext: func(_ context.Context, _, _ string) (net.Conn, error) {
|
|
|
|
return net.Dial("unix", conf.Global.Control.Sockpath)
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}, nil
|
2017-09-17 23:54:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func doControlPProf(cmd *cobra.Command, args []string) {
|
|
|
|
|
|
|
|
log := golog.New(os.Stderr, "", 0)
|
|
|
|
|
|
|
|
die := func() {
|
|
|
|
log.Printf("exiting after error")
|
|
|
|
os.Exit(1)
|
|
|
|
}
|
|
|
|
|
|
|
|
if cmd.Flags().Arg(0) != "cpu" {
|
|
|
|
log.Printf("only CPU profiles are supported")
|
|
|
|
log.Printf("%s", cmd.UsageString())
|
|
|
|
die()
|
|
|
|
}
|
|
|
|
|
|
|
|
outfn := cmd.Flags().Arg(1)
|
|
|
|
if outfn == "" {
|
|
|
|
log.Printf("must specify output filename")
|
|
|
|
log.Printf("%s", cmd.UsageString())
|
|
|
|
die()
|
|
|
|
}
|
|
|
|
var out io.Writer
|
2017-11-18 20:34:28 +01:00
|
|
|
var err error
|
2017-09-17 23:54:23 +02:00
|
|
|
if outfn == "-" {
|
|
|
|
out = os.Stdout
|
|
|
|
} else {
|
|
|
|
out, err = os.Create(outfn)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("error creating output file: %s", err)
|
|
|
|
die()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Printf("connecting to daemon")
|
2017-11-18 20:34:28 +01:00
|
|
|
httpc, err := controlHttpClient()
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("error parsing config: %s", err)
|
|
|
|
die()
|
2017-09-17 23:54:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
log.Printf("profiling...")
|
|
|
|
v := url.Values{}
|
|
|
|
v.Set("seconds", fmt.Sprintf("%d", pprofCmdArgs.seconds))
|
|
|
|
v.Encode()
|
|
|
|
resp, err := httpc.Get("http://unix" + ControlJobEndpointProfile + "?" + v.Encode())
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("error: %s", err)
|
|
|
|
die()
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err = io.Copy(out, resp.Body)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("error writing profile: %s", err)
|
|
|
|
die()
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Printf("finished")
|
|
|
|
|
|
|
|
}
|
2017-11-18 20:34:28 +01:00
|
|
|
|
|
|
|
func doControLVersionCmd(cmd *cobra.Command, args []string) {
|
|
|
|
|
|
|
|
log := golog.New(os.Stderr, "", 0)
|
|
|
|
|
|
|
|
die := func() {
|
|
|
|
log.Printf("exiting after error")
|
|
|
|
os.Exit(1)
|
|
|
|
}
|
|
|
|
|
|
|
|
httpc, err := controlHttpClient()
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("could not connect to daemon: %s", err)
|
|
|
|
die()
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, err := httpc.Get("http://unix" + ControlJobEndpointVersion)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("error: %s", err)
|
|
|
|
die()
|
|
|
|
} else if resp.StatusCode != http.StatusOK {
|
|
|
|
var msg bytes.Buffer
|
|
|
|
io.CopyN(&msg, resp.Body, 4096)
|
|
|
|
log.Printf("error: %s", msg.String())
|
|
|
|
die()
|
|
|
|
}
|
|
|
|
|
|
|
|
var info ZreplVersionInformation
|
|
|
|
err = json.NewDecoder(resp.Body).Decode(&info)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("error unmarshaling response: %s", err)
|
|
|
|
die()
|
|
|
|
}
|
|
|
|
|
|
|
|
fmt.Println(info.String())
|
|
|
|
|
|
|
|
}
|