mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2025-01-02 09:10:40 +01:00
488c34f5e1
vmctl: fix vmctl blocking on process interrupt This change prevents vmctl from indefinite blocking on receiving the interrupt signal. The update touches all import modes and suppose to improve tool reliability. https://github.com/VictoriaMetrics/VictoriaMetrics/issues/2491
159 lines
3.9 KiB
Go
159 lines
3.9 KiB
Go
package main
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"log"
|
|
"net/http"
|
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmctl/barpool"
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmctl/limiter"
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmctl/vm"
|
|
)
|
|
|
|
type vmNativeProcessor struct {
|
|
filter filter
|
|
rateLimit int64
|
|
|
|
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"
|
|
|
|
nativeBarTpl = `Total: {{counters . }} {{ cycle . "↖" "↗" "↘" "↙" }} Speed: {{speed . }} {{string . "suffix"}}`
|
|
)
|
|
|
|
func (p *vmNativeProcessor) run(ctx context.Context) 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(ctx)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to init export pipe: %s", err)
|
|
}
|
|
|
|
nativeImportAddr, err := vm.AddExtraLabelsToImportPath(nativeImportAddr, p.dst.extraLabels)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
sync := make(chan struct{})
|
|
go func() {
|
|
defer func() { close(sync) }()
|
|
u := fmt.Sprintf("%s/%s", p.dst.addr, nativeImportAddr)
|
|
req, err := http.NewRequestWithContext(ctx, "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)
|
|
}
|
|
if err := importResp.Body.Close(); err != nil {
|
|
log.Fatalf("cannot close import response body: %s", err)
|
|
}
|
|
}()
|
|
|
|
fmt.Printf("Initing import process to %q:\n", p.dst.addr)
|
|
bar := barpool.AddWithTemplate(nativeBarTpl, 0)
|
|
barReader := bar.NewProxyReader(exportReader)
|
|
if err := barpool.Start(); err != nil {
|
|
log.Printf("error start process bars pool: %s", err)
|
|
return err
|
|
}
|
|
|
|
w := io.Writer(pw)
|
|
if p.rateLimit > 0 {
|
|
rl := limiter.NewLimiter(p.rateLimit)
|
|
w = limiter.NewWriteLimiter(pw, rl)
|
|
}
|
|
|
|
_, err = io.Copy(w, 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
|
|
|
|
barpool.Stop()
|
|
log.Println("Import finished!")
|
|
return nil
|
|
}
|
|
|
|
func (p *vmNativeProcessor) exportPipe(ctx context.Context) (io.ReadCloser, error) {
|
|
u := fmt.Sprintf("%s/%s", p.src.addr, nativeExportAddr)
|
|
req, err := http.NewRequestWithContext(ctx, "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
|
|
}
|