From bea5a8700ad979f65711207846abee4c0b5af57d Mon Sep 17 00:00:00 2001 From: Aliaksandr Valialkin Date: Fri, 20 Dec 2019 11:34:52 +0200 Subject: [PATCH] app/vmselect: add `-search.maxExportDuration` command-line flag for limiting `/api/v1/export` duration Updates https://github.com/VictoriaMetrics/VictoriaMetrics/issues/275 --- app/vmselect/prometheus/prometheus.go | 38 +++++++++++++++++---------- 1 file changed, 24 insertions(+), 14 deletions(-) diff --git a/app/vmselect/prometheus/prometheus.go b/app/vmselect/prometheus/prometheus.go index b509fc002c..35797ba1af 100644 --- a/app/vmselect/prometheus/prometheus.go +++ b/app/vmselect/prometheus/prometheus.go @@ -23,9 +23,10 @@ import ( var ( latencyOffset = flag.Duration("search.latencyOffset", time.Second*30, "The time when data points become visible in query results after the colection. "+ "Too small value can result in incomplete last points for query results") - maxQueryDuration = flag.Duration("search.maxQueryDuration", time.Second*30, "The maximum time for search query execution") - maxQueryLen = flag.Int("search.maxQueryLen", 16*1024, "The maximum search query length in bytes") - maxLookback = flag.Duration("search.maxLookback", 0, "Synonim to `-search.lookback-delta` from Prometheus. "+ + maxExportDuration = flag.Duration("search.maxExportDuration", 10*time.Minute, "The maximum duration for `/api/v1/export` call") + maxQueryDuration = flag.Duration("search.maxQueryDuration", time.Second*30, "The maximum duration for search query execution") + maxQueryLen = flag.Int("search.maxQueryLen", 16*1024, "The maximum search query length in bytes") + maxLookback = flag.Duration("search.maxLookback", 0, "Synonim to `-search.lookback-delta` from Prometheus. "+ "The value is dynamically detected from interval between time series datapoints if not set. It can be overridden on per-query basis via `max_lookback` arg") ) @@ -58,7 +59,7 @@ func FederateHandler(w http.ResponseWriter, r *http.Request) error { if err != nil { return err } - deadline := getDeadline(r) + deadline := getDeadlineForQuery(r) if start >= end { start = end - defaultStep } @@ -129,7 +130,7 @@ func ExportHandler(w http.ResponseWriter, r *http.Request) error { return err } format := r.FormValue("format") - deadline := getDeadline(r) + deadline := getDeadlineForExport(r) if start >= end { end = start + defaultStep } @@ -235,7 +236,7 @@ var deleteDuration = metrics.NewSummary(`vm_request_duration_seconds{path="/api/ // See https://prometheus.io/docs/prometheus/latest/querying/api/#querying-label-values func LabelValuesHandler(labelName string, w http.ResponseWriter, r *http.Request) error { startTime := time.Now() - deadline := getDeadline(r) + deadline := getDeadlineForQuery(r) if err := r.ParseForm(); err != nil { return fmt.Errorf("cannot parse form values: %s", err) @@ -333,7 +334,7 @@ var labelValuesDuration = metrics.NewSummary(`vm_request_duration_seconds{path=" // LabelsCountHandler processes /api/v1/labels/count request. func LabelsCountHandler(w http.ResponseWriter, r *http.Request) error { startTime := time.Now() - deadline := getDeadline(r) + deadline := getDeadlineForQuery(r) labelEntries, err := netstorage.GetLabelEntries(deadline) if err != nil { return fmt.Errorf(`cannot obtain label entries: %s`, err) @@ -352,7 +353,7 @@ var labelsCountDuration = metrics.NewSummary(`vm_request_duration_seconds{path=" // See https://prometheus.io/docs/prometheus/latest/querying/api/#getting-label-names func LabelsHandler(w http.ResponseWriter, r *http.Request) error { startTime := time.Now() - deadline := getDeadline(r) + deadline := getDeadlineForQuery(r) if err := r.ParseForm(); err != nil { return fmt.Errorf("cannot parse form values: %s", err) @@ -442,7 +443,7 @@ var labelsDuration = metrics.NewSummary(`vm_request_duration_seconds{path="/api/ // SeriesCountHandler processes /api/v1/series/count request. func SeriesCountHandler(w http.ResponseWriter, r *http.Request) error { startTime := time.Now() - deadline := getDeadline(r) + deadline := getDeadlineForQuery(r) n, err := netstorage.GetSeriesCount(deadline) if err != nil { return fmt.Errorf("cannot obtain series count: %s", err) @@ -482,7 +483,7 @@ func SeriesHandler(w http.ResponseWriter, r *http.Request) error { if err != nil { return err } - deadline := getDeadline(r) + deadline := getDeadlineForQuery(r) tagFilterss, err := getTagFilterssFromMatches(matches) if err != nil { @@ -551,7 +552,7 @@ func QueryHandler(w http.ResponseWriter, r *http.Request) error { if err != nil { return err } - deadline := getDeadline(r) + deadline := getDeadlineForQuery(r) lookbackDelta, err := getMaxLookback(r) if err != nil { return err @@ -674,7 +675,7 @@ func QueryRangeHandler(w http.ResponseWriter, r *http.Request) error { } func queryRangeHandler(w http.ResponseWriter, query string, start, end, step int64, r *http.Request, ct int64) error { - deadline := getDeadline(r) + deadline := getDeadlineForQuery(r) mayCache := !getBool(r, "nocache") lookbackDelta, err := getMaxLookback(r) if err != nil { @@ -866,12 +867,21 @@ func getMaxLookback(r *http.Request) (int64, error) { return getDuration(r, "max_lookback", d) } -func getDeadline(r *http.Request) netstorage.Deadline { +func getDeadlineForQuery(r *http.Request) netstorage.Deadline { + dMax := int64(maxQueryDuration.Seconds() * 1e3) + return getDeadlineWithMaxDuration(r, dMax) +} + +func getDeadlineForExport(r *http.Request) netstorage.Deadline { + dMax := int64(maxExportDuration.Seconds() * 1e3) + return getDeadlineWithMaxDuration(r, dMax) +} + +func getDeadlineWithMaxDuration(r *http.Request, dMax int64) netstorage.Deadline { d, err := getDuration(r, "timeout", 0) if err != nil { d = 0 } - dMax := int64(maxQueryDuration.Seconds() * 1e3) if d <= 0 || d > dMax { d = dMax }