mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2025-01-04 13:52:05 +01:00
fc5292d8ed
* app/vmctl: add verbose output for docker installations or when TTY isn't available * app/vmctl: fix tests * app/vmctl: make vmctl interactive if no tty * app/vmctl: cleanup * app/vmctl: add comment --------- Co-authored-by: Nikolay <nik@victoriametrics.com>
130 lines
3.0 KiB
Go
130 lines
3.0 KiB
Go
package main
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"log"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmctl/barpool"
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmctl/remoteread"
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmctl/stepper"
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmctl/vm"
|
|
"github.com/cheggaaa/pb/v3"
|
|
)
|
|
|
|
type remoteReadProcessor struct {
|
|
filter remoteReadFilter
|
|
|
|
dst *vm.Importer
|
|
src *remoteread.Client
|
|
|
|
cc int
|
|
isSilent bool
|
|
isVerbose bool
|
|
}
|
|
|
|
type remoteReadFilter struct {
|
|
timeStart *time.Time
|
|
timeEnd *time.Time
|
|
chunk string
|
|
}
|
|
|
|
func (rrp *remoteReadProcessor) run(ctx context.Context) error {
|
|
rrp.dst.ResetStats()
|
|
if rrp.filter.timeEnd == nil {
|
|
t := time.Now().In(rrp.filter.timeStart.Location())
|
|
rrp.filter.timeEnd = &t
|
|
}
|
|
if rrp.cc < 1 {
|
|
rrp.cc = 1
|
|
}
|
|
|
|
ranges, err := stepper.SplitDateRange(*rrp.filter.timeStart, *rrp.filter.timeEnd, rrp.filter.chunk)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to create date ranges for the given time filters: %v", err)
|
|
}
|
|
|
|
question := fmt.Sprintf("Selected time range %q - %q will be split into %d ranges according to %q step. Continue?",
|
|
rrp.filter.timeStart.String(), rrp.filter.timeEnd.String(), len(ranges), rrp.filter.chunk)
|
|
if !rrp.isSilent && !prompt(question) {
|
|
return nil
|
|
}
|
|
|
|
var bar *pb.ProgressBar
|
|
if !rrp.isSilent {
|
|
bar = barpool.AddWithTemplate(fmt.Sprintf(barTpl, "Processing ranges"), len(ranges))
|
|
if err := barpool.Start(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
defer func() {
|
|
if !rrp.isSilent {
|
|
barpool.Stop()
|
|
}
|
|
log.Println("Import finished!")
|
|
log.Print(rrp.dst.Stats())
|
|
}()
|
|
|
|
rangeC := make(chan *remoteread.Filter)
|
|
errCh := make(chan error)
|
|
|
|
var wg sync.WaitGroup
|
|
wg.Add(rrp.cc)
|
|
for i := 0; i < rrp.cc; i++ {
|
|
go func() {
|
|
defer wg.Done()
|
|
for r := range rangeC {
|
|
if err := rrp.do(ctx, r); err != nil {
|
|
errCh <- fmt.Errorf("request failed for: %s", err)
|
|
return
|
|
}
|
|
if bar != nil {
|
|
bar.Increment()
|
|
}
|
|
}
|
|
}()
|
|
}
|
|
|
|
for _, r := range ranges {
|
|
select {
|
|
case infErr := <-errCh:
|
|
return fmt.Errorf("remote read error: %s", infErr)
|
|
case vmErr := <-rrp.dst.Errors():
|
|
return fmt.Errorf("import process failed: %s", wrapErr(vmErr, rrp.isVerbose))
|
|
case rangeC <- &remoteread.Filter{
|
|
StartTimestampMs: r[0].UnixMilli(),
|
|
EndTimestampMs: r[1].UnixMilli(),
|
|
}:
|
|
}
|
|
}
|
|
|
|
close(rangeC)
|
|
wg.Wait()
|
|
rrp.dst.Close()
|
|
close(errCh)
|
|
// drain import errors channel
|
|
for vmErr := range rrp.dst.Errors() {
|
|
if vmErr.Err != nil {
|
|
return fmt.Errorf("import process failed: %s", wrapErr(vmErr, rrp.isVerbose))
|
|
}
|
|
}
|
|
for err := range errCh {
|
|
return fmt.Errorf("import process failed: %s", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (rrp *remoteReadProcessor) do(ctx context.Context, filter *remoteread.Filter) error {
|
|
return rrp.src.Read(ctx, filter, func(series *vm.TimeSeries) error {
|
|
if err := rrp.dst.Input(series); err != nil {
|
|
return fmt.Errorf(
|
|
"failed to read data for time range start: %d, end: %d, %s",
|
|
filter.StartTimestampMs, filter.EndTimestampMs, err)
|
|
}
|
|
return nil
|
|
})
|
|
}
|