mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2025-01-20 15:16:42 +00:00
25e54d2b50
add progress bars to the VM importer The new progress bars supposed to display the processing speed per each VM importer worker. This info should help to identify if there is a bottleneck on the VM side during the import process, without waiting for its finish. The new progress bars can be disabled by passing `vm-disable-progress-bar` flag. Plotting multiple progress bars requires using experimental progress bar pool from github.com/cheggaaa/pb/v3. Switch to progress bar pool required changes in all import modes. The openTSDB mode wasn't changed due to its implementation, which implies individual progress bars per each series. Because of this, using the pool wasn't possible. Signed-off-by: dmitryk-dk <kozlovdmitriyy@gmail.com> Co-authored-by: hagen1778 <roman@victoriametrics.com>
154 lines
3.8 KiB
Go
154 lines
3.8 KiB
Go
package main
|
|
|
|
import (
|
|
"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() 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)
|
|
}
|
|
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()
|
|
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
|
|
}
|