mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2024-11-21 14:44:00 +00:00
d5c180e680
It is better developing vmctl tool in VictoriaMetrics repository, so it could be released together with the rest of vmutils tools such as vmalert, vmagent, vmbackup, vmrestore and vmauth.
141 lines
3.4 KiB
Go
141 lines
3.4 KiB
Go
package main
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"log"
|
|
"net/http"
|
|
|
|
"github.com/cheggaaa/pb/v3"
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmctl/vm"
|
|
)
|
|
|
|
type vmNativeProcessor struct {
|
|
filter filter
|
|
|
|
dst *vmNativeClient
|
|
src *vmNativeClient
|
|
}
|
|
|
|
type vmNativeClient struct {
|
|
addr string
|
|
user string
|
|
password string
|
|
extraLabels []string
|
|
}
|
|
|
|
type filter struct {
|
|
match string
|
|
timeStart string
|
|
timeEnd string
|
|
}
|
|
|
|
func (f filter) String() string {
|
|
s := fmt.Sprintf("\n\tfilter: match[]=%s", f.match)
|
|
if f.timeStart != "" {
|
|
s += fmt.Sprintf("\n\tstart: %s", f.timeStart)
|
|
}
|
|
if f.timeEnd != "" {
|
|
s += fmt.Sprintf("\n\tend: %s", f.timeEnd)
|
|
}
|
|
return s
|
|
}
|
|
|
|
const (
|
|
nativeExportAddr = "api/v1/export/native"
|
|
nativeImportAddr = "api/v1/import/native"
|
|
|
|
barTpl = `Total: {{counters . }} {{ cycle . "↖" "↗" "↘" "↙" }} Speed: {{speed . }} {{string . "suffix"}}`
|
|
)
|
|
|
|
func (p *vmNativeProcessor) run() error {
|
|
pr, pw := io.Pipe()
|
|
|
|
fmt.Printf("Initing export pipe from %q with filters: %s\n", p.src.addr, p.filter)
|
|
exportReader, err := p.exportPipe()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to init export pipe: %s", err)
|
|
}
|
|
|
|
sync := make(chan struct{})
|
|
nativeImportAddr, err := vm.AddExtraLabelsToImportPath(nativeImportAddr, p.dst.extraLabels)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
go func() {
|
|
defer func() { close(sync) }()
|
|
u := fmt.Sprintf("%s/%s", p.dst.addr, nativeImportAddr)
|
|
req, err := http.NewRequest("POST", u, pr)
|
|
if err != nil {
|
|
log.Fatalf("cannot create import request to %q: %s", p.dst.addr, err)
|
|
}
|
|
importResp, err := p.dst.do(req, http.StatusNoContent)
|
|
if err != nil {
|
|
log.Fatalf("import request failed: %s", err)
|
|
}
|
|
importResp.Body.Close()
|
|
}()
|
|
|
|
fmt.Printf("Initing import process to %q:\n", p.dst.addr)
|
|
bar := pb.ProgressBarTemplate(barTpl).Start64(0)
|
|
barReader := bar.NewProxyReader(exportReader)
|
|
|
|
_, err = io.Copy(pw, barReader)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to write into %q: %s", p.dst.addr, err)
|
|
}
|
|
if err := pw.Close(); err != nil {
|
|
return err
|
|
}
|
|
<-sync
|
|
|
|
bar.Finish()
|
|
return nil
|
|
}
|
|
|
|
func (p *vmNativeProcessor) exportPipe() (io.ReadCloser, error) {
|
|
u := fmt.Sprintf("%s/%s", p.src.addr, nativeExportAddr)
|
|
req, err := http.NewRequest("GET", u, nil)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("cannot create request to %q: %s", p.src.addr, err)
|
|
}
|
|
|
|
params := req.URL.Query()
|
|
params.Set("match[]", p.filter.match)
|
|
if p.filter.timeStart != "" {
|
|
params.Set("start", p.filter.timeStart)
|
|
}
|
|
if p.filter.timeEnd != "" {
|
|
params.Set("end", p.filter.timeEnd)
|
|
}
|
|
req.URL.RawQuery = params.Encode()
|
|
|
|
// disable compression since it is meaningless for native format
|
|
req.Header.Set("Accept-Encoding", "identity")
|
|
resp, err := p.src.do(req, http.StatusOK)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("export request failed: %s", err)
|
|
}
|
|
return resp.Body, nil
|
|
}
|
|
|
|
func (c *vmNativeClient) do(req *http.Request, expSC int) (*http.Response, error) {
|
|
if c.user != "" {
|
|
req.SetBasicAuth(c.user, c.password)
|
|
}
|
|
resp, err := http.DefaultClient.Do(req)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("unexpected error when performing request: %s", err)
|
|
}
|
|
|
|
if resp.StatusCode != expSC {
|
|
body, err := ioutil.ReadAll(resp.Body)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to read response body for status code %d: %s", resp.StatusCode, err)
|
|
}
|
|
return nil, fmt.Errorf("unexpected response code %d: %s", resp.StatusCode, string(body))
|
|
}
|
|
return resp, err
|
|
}
|