diff --git a/app/vmselect/graphite/tags_api.go b/app/vmselect/graphite/tags_api.go index de7416bb4..6b3b9f613 100644 --- a/app/vmselect/graphite/tags_api.go +++ b/app/vmselect/graphite/tags_api.go @@ -193,7 +193,7 @@ func TagsAutoCompleteValuesHandler(startTime time.Time, at *auth.Token, w http.R if err != nil { return err } - mns, isPartialResponse, err := netstorage.SearchMetricNames(nil, at, denyPartialResponse, sq, deadline) + metricNames, isPartialResponse, err := netstorage.SearchMetricNames(nil, at, denyPartialResponse, sq, deadline) if err != nil { return fmt.Errorf("cannot fetch metric names for %q: %w", sq, err) } @@ -202,7 +202,11 @@ func TagsAutoCompleteValuesHandler(startTime time.Time, at *auth.Token, w http.R if tag == "name" { tag = "__name__" } - for _, mn := range mns { + var mn storage.MetricName + for _, metricName := range metricNames { + if err := mn.UnmarshalString(metricName); err != nil { + return fmt.Errorf("cannot unmarshal metricName=%q: %w", metricName, err) + } tagValue := mn.GetTagValue(tag) if len(tagValue) == 0 { continue @@ -279,13 +283,17 @@ func TagsAutoCompleteTagsHandler(startTime time.Time, at *auth.Token, w http.Res if err != nil { return err } - mns, isPartialResponse, err := netstorage.SearchMetricNames(nil, at, denyPartialResponse, sq, deadline) + metricNames, isPartialResponse, err := netstorage.SearchMetricNames(nil, at, denyPartialResponse, sq, deadline) if err != nil { return fmt.Errorf("cannot fetch metric names for %q: %w", sq, err) } isPartial = isPartialResponse m := make(map[string]struct{}) - for _, mn := range mns { + var mn storage.MetricName + for _, metricName := range metricNames { + if err := mn.UnmarshalString(metricName); err != nil { + return fmt.Errorf("cannot unmarshal metricName=%q: %w", metricName, err) + } m["name"] = struct{}{} for _, tag := range mn.Tags { m[string(tag.Key)] = struct{}{} @@ -345,11 +353,14 @@ func TagsFindSeriesHandler(startTime time.Time, at *auth.Token, w http.ResponseW return err } denyPartialResponse := searchutils.GetDenyPartialResponse(r) - mns, isPartial, err := netstorage.SearchMetricNames(nil, at, denyPartialResponse, sq, deadline) + metricNames, isPartial, err := netstorage.SearchMetricNames(nil, at, denyPartialResponse, sq, deadline) if err != nil { return fmt.Errorf("cannot fetch metric names for %q: %w", sq, err) } - paths := getCanonicalPaths(mns) + paths, err := getCanonicalPaths(metricNames) + if err != nil { + return fmt.Errorf("cannot obtain canonical paths: %w", err) + } if limit > 0 && limit < len(paths) { paths = paths[:limit] } @@ -365,14 +376,18 @@ func TagsFindSeriesHandler(startTime time.Time, at *auth.Token, w http.ResponseW return nil } -func getCanonicalPaths(mns []storage.MetricName) []string { - paths := make([]string, 0, len(mns)) - for _, mn := range mns { +func getCanonicalPaths(metricNames []string) ([]string, error) { + paths := make([]string, 0, len(metricNames)) + var mn storage.MetricName + for _, metricName := range metricNames { + if err := mn.UnmarshalString(metricName); err != nil { + return nil, fmt.Errorf("cannot unmarshal metricName=%q: %w", metricName, err) + } path := getCanonicalPath(&mn) paths = append(paths, path) } sort.Strings(paths) - return paths + return paths, nil } func getCanonicalPath(mn *storage.MetricName) string { diff --git a/app/vmselect/netstorage/netstorage.go b/app/vmselect/netstorage/netstorage.go index 6501de6f3..4ce54393c 100644 --- a/app/vmselect/netstorage/netstorage.go +++ b/app/vmselect/netstorage/netstorage.go @@ -1176,7 +1176,9 @@ func ExportBlocks(qt *querytracer.Tracer, at *auth.Token, sq *storage.SearchQuer } // SearchMetricNames returns all the metric names matching sq until the given deadline. -func SearchMetricNames(qt *querytracer.Tracer, at *auth.Token, denyPartialResponse bool, sq *storage.SearchQuery, deadline searchutils.Deadline) ([]storage.MetricName, bool, error) { +// +// The returned metric names must be unmarshaled via storage.MetricName.UnmarshalString(). +func SearchMetricNames(qt *querytracer.Tracer, at *auth.Token, denyPartialResponse bool, sq *storage.SearchQuery, deadline searchutils.Deadline) ([]string, bool, error) { qt = qt.NewChild("fetch metric names: %s", sq) defer qt.Done() if deadline.Exceeded() { @@ -1186,7 +1188,7 @@ func SearchMetricNames(qt *querytracer.Tracer, at *auth.Token, denyPartialRespon // Send the query to all the storage nodes in parallel. type nodeResult struct { - metricNames [][]byte + metricNames []string err error } snr := startStorageNodesRequest(qt, denyPartialResponse, func(qt *querytracer.Tracer, idx int, sn *storageNode) interface{} { @@ -1203,14 +1205,14 @@ func SearchMetricNames(qt *querytracer.Tracer, at *auth.Token, denyPartialRespon }) // Collect results. - metricNames := make(map[string]struct{}) + metricNamesMap := make(map[string]struct{}) isPartial, err := snr.collectResults(partialSearchMetricNamesResults, func(result interface{}) error { nr := result.(*nodeResult) if nr.err != nil { return nr.err } for _, metricName := range nr.metricNames { - metricNames[string(metricName)] = struct{}{} + metricNamesMap[metricName] = struct{}{} } return nil }) @@ -1218,17 +1220,13 @@ func SearchMetricNames(qt *querytracer.Tracer, at *auth.Token, denyPartialRespon return nil, isPartial, fmt.Errorf("cannot fetch metric names from vmstorage nodes: %w", err) } - // Unmarshal metricNames - mns := make([]storage.MetricName, len(metricNames)) - i := 0 - for metricName := range metricNames { - mn := &mns[i] - if err := mn.Unmarshal(bytesutil.ToUnsafeBytes(metricName)); err != nil { - return nil, false, fmt.Errorf("cannot unmarshal metric name obtained from vmstorage: %w; metricName=%q", err, metricName) - } - i++ + metricNames := make([]string, len(metricNamesMap)) + for metricName := range metricNamesMap { + metricNames = append(metricNames, metricName) } - return mns, isPartial, nil + sort.Strings(metricNames) + qt.Printf("sort %d metric names", len(metricNames)) + return metricNames, isPartial, nil } // ProcessSearchQuery performs sq until the given deadline. @@ -1592,8 +1590,8 @@ func (sn *storageNode) getSeriesCount(qt *querytracer.Tracer, accountID, project return n, nil } -func (sn *storageNode) processSearchMetricNames(qt *querytracer.Tracer, requestData []byte, deadline searchutils.Deadline) ([][]byte, error) { - var metricNames [][]byte +func (sn *storageNode) processSearchMetricNames(qt *querytracer.Tracer, requestData []byte, deadline searchutils.Deadline) ([]string, error) { + var metricNames []string f := func(bc *handshake.BufferedConn) error { mns, err := sn.processSearchMetricNamesOnConn(bc, requestData) if err != nil { @@ -2071,7 +2069,7 @@ const maxMetricBlockSize = 1024 * 1024 // from vmstorage. const maxErrorMessageSize = 64 * 1024 -func (sn *storageNode) processSearchMetricNamesOnConn(bc *handshake.BufferedConn, requestData []byte) ([][]byte, error) { +func (sn *storageNode) processSearchMetricNamesOnConn(bc *handshake.BufferedConn, requestData []byte) ([]string, error) { // Send the requst to sn. if err := writeBytes(bc, requestData); err != nil { return nil, fmt.Errorf("cannot write requestData: %w", err) @@ -2094,13 +2092,13 @@ func (sn *storageNode) processSearchMetricNamesOnConn(bc *handshake.BufferedConn if err != nil { return nil, fmt.Errorf("cannot read metricNamesCount: %w", err) } - metricNames := make([][]byte, metricNamesCount) + metricNames := make([]string, metricNamesCount) for i := int64(0); i < int64(metricNamesCount); i++ { buf, err = readBytes(buf[:0], bc, maxMetricNameSize) if err != nil { return nil, fmt.Errorf("cannot read metricName #%d: %w", i+1, err) } - metricNames[i] = append(metricNames[i][:0], buf...) + metricNames[i] = string(buf) } return metricNames, nil } diff --git a/app/vmselect/prometheus/prometheus.go b/app/vmselect/prometheus/prometheus.go index af72f6089..361b7aba5 100644 --- a/app/vmselect/prometheus/prometheus.go +++ b/app/vmselect/prometheus/prometheus.go @@ -675,7 +675,7 @@ func SeriesHandler(qt *querytracer.Tracer, startTime time.Time, at *auth.Token, } sq := storage.NewSearchQuery(at.AccountID, at.ProjectID, cp.start, cp.end, cp.filterss, *maxSeriesLimit) denyPartialResponse := searchutils.GetDenyPartialResponse(r) - mns, isPartial, err := netstorage.SearchMetricNames(qt, at, denyPartialResponse, sq, cp.deadline) + metricNames, isPartial, err := netstorage.SearchMetricNames(qt, at, denyPartialResponse, sq, cp.deadline) if err != nil { return fmt.Errorf("cannot fetch time series for %q: %w", sq, err) } @@ -685,7 +685,7 @@ func SeriesHandler(qt *querytracer.Tracer, startTime time.Time, at *auth.Token, qtDone := func() { qt.Donef("start=%d, end=%d", cp.start, cp.end) } - WriteSeriesResponse(bw, isPartial, mns, qt, qtDone) + WriteSeriesResponse(bw, isPartial, metricNames, qt, qtDone) if err := bw.Flush(); err != nil { return err } diff --git a/app/vmselect/prometheus/series_response.qtpl b/app/vmselect/prometheus/series_response.qtpl index 0cb7f9aef..6ebbb1f08 100644 --- a/app/vmselect/prometheus/series_response.qtpl +++ b/app/vmselect/prometheus/series_response.qtpl @@ -6,18 +6,24 @@ {% stripspace %} SeriesResponse generates response for /api/v1/series. See https://prometheus.io/docs/prometheus/latest/querying/api/#finding-series-by-label-matchers -{% func SeriesResponse(isPartial bool, mns []storage.MetricName, qt *querytracer.Tracer, qtDone func()) %} +{% func SeriesResponse(isPartial bool, metricNames []string, qt *querytracer.Tracer, qtDone func()) %} { "status":"success", "isPartial":{% if isPartial %}true{% else %}false{% endif %}, "data":[ - {% for i := range mns %} - {%= metricNameObject(&mns[i]) %} - {% if i+1 < len(mns) %},{% endif %} + {% code var mn storage.MetricName %} + {% for i, metricName := range metricNames %} + {% code err := mn.UnmarshalString(metricName) %} + {% if err != nil %} + {%q= err.Error() %} + {% else %} + {%= metricNameObject(&mn) %} + {% endif %} + {% if i+1 < len(metricNames) %},{% endif %} {% endfor %} ] {% code - qt.Printf("generate response: series=%d", len(mns)) + qt.Printf("generate response: series=%d", len(metricNames)) qtDone() %} {%= dumpQueryTrace(qt) %} diff --git a/app/vmselect/prometheus/series_response.qtpl.go b/app/vmselect/prometheus/series_response.qtpl.go index ff49a935d..ee79b7094 100644 --- a/app/vmselect/prometheus/series_response.qtpl.go +++ b/app/vmselect/prometheus/series_response.qtpl.go @@ -26,7 +26,7 @@ var ( ) //line app/vmselect/prometheus/series_response.qtpl:9 -func StreamSeriesResponse(qw422016 *qt422016.Writer, isPartial bool, mns []storage.MetricName, qt *querytracer.Tracer, qtDone func()) { +func StreamSeriesResponse(qw422016 *qt422016.Writer, isPartial bool, metricNames []string, qt *querytracer.Tracer, qtDone func()) { //line app/vmselect/prometheus/series_response.qtpl:9 qw422016.N().S(`{"status":"success","isPartial":`) //line app/vmselect/prometheus/series_response.qtpl:12 @@ -42,52 +42,66 @@ func StreamSeriesResponse(qw422016 *qt422016.Writer, isPartial bool, mns []stora //line app/vmselect/prometheus/series_response.qtpl:12 qw422016.N().S(`,"data":[`) //line app/vmselect/prometheus/series_response.qtpl:14 - for i := range mns { + var mn storage.MetricName + //line app/vmselect/prometheus/series_response.qtpl:15 - streammetricNameObject(qw422016, &mns[i]) + for i, metricName := range metricNames { //line app/vmselect/prometheus/series_response.qtpl:16 - if i+1 < len(mns) { -//line app/vmselect/prometheus/series_response.qtpl:16 - qw422016.N().S(`,`) -//line app/vmselect/prometheus/series_response.qtpl:16 - } + err := mn.UnmarshalString(metricName) + //line app/vmselect/prometheus/series_response.qtpl:17 - } -//line app/vmselect/prometheus/series_response.qtpl:17 - qw422016.N().S(`]`) + if err != nil { +//line app/vmselect/prometheus/series_response.qtpl:18 + qw422016.N().Q(err.Error()) +//line app/vmselect/prometheus/series_response.qtpl:19 + } else { //line app/vmselect/prometheus/series_response.qtpl:20 - qt.Printf("generate response: series=%d", len(mns)) + streammetricNameObject(qw422016, &mn) +//line app/vmselect/prometheus/series_response.qtpl:21 + } +//line app/vmselect/prometheus/series_response.qtpl:22 + if i+1 < len(metricNames) { +//line app/vmselect/prometheus/series_response.qtpl:22 + qw422016.N().S(`,`) +//line app/vmselect/prometheus/series_response.qtpl:22 + } +//line app/vmselect/prometheus/series_response.qtpl:23 + } +//line app/vmselect/prometheus/series_response.qtpl:23 + qw422016.N().S(`]`) +//line app/vmselect/prometheus/series_response.qtpl:26 + qt.Printf("generate response: series=%d", len(metricNames)) qtDone() -//line app/vmselect/prometheus/series_response.qtpl:23 +//line app/vmselect/prometheus/series_response.qtpl:29 streamdumpQueryTrace(qw422016, qt) -//line app/vmselect/prometheus/series_response.qtpl:23 +//line app/vmselect/prometheus/series_response.qtpl:29 qw422016.N().S(`}`) -//line app/vmselect/prometheus/series_response.qtpl:25 +//line app/vmselect/prometheus/series_response.qtpl:31 } -//line app/vmselect/prometheus/series_response.qtpl:25 -func WriteSeriesResponse(qq422016 qtio422016.Writer, isPartial bool, mns []storage.MetricName, qt *querytracer.Tracer, qtDone func()) { -//line app/vmselect/prometheus/series_response.qtpl:25 +//line app/vmselect/prometheus/series_response.qtpl:31 +func WriteSeriesResponse(qq422016 qtio422016.Writer, isPartial bool, metricNames []string, qt *querytracer.Tracer, qtDone func()) { +//line app/vmselect/prometheus/series_response.qtpl:31 qw422016 := qt422016.AcquireWriter(qq422016) -//line app/vmselect/prometheus/series_response.qtpl:25 - StreamSeriesResponse(qw422016, isPartial, mns, qt, qtDone) -//line app/vmselect/prometheus/series_response.qtpl:25 +//line app/vmselect/prometheus/series_response.qtpl:31 + StreamSeriesResponse(qw422016, isPartial, metricNames, qt, qtDone) +//line app/vmselect/prometheus/series_response.qtpl:31 qt422016.ReleaseWriter(qw422016) -//line app/vmselect/prometheus/series_response.qtpl:25 +//line app/vmselect/prometheus/series_response.qtpl:31 } -//line app/vmselect/prometheus/series_response.qtpl:25 -func SeriesResponse(isPartial bool, mns []storage.MetricName, qt *querytracer.Tracer, qtDone func()) string { -//line app/vmselect/prometheus/series_response.qtpl:25 +//line app/vmselect/prometheus/series_response.qtpl:31 +func SeriesResponse(isPartial bool, metricNames []string, qt *querytracer.Tracer, qtDone func()) string { +//line app/vmselect/prometheus/series_response.qtpl:31 qb422016 := qt422016.AcquireByteBuffer() -//line app/vmselect/prometheus/series_response.qtpl:25 - WriteSeriesResponse(qb422016, isPartial, mns, qt, qtDone) -//line app/vmselect/prometheus/series_response.qtpl:25 +//line app/vmselect/prometheus/series_response.qtpl:31 + WriteSeriesResponse(qb422016, isPartial, metricNames, qt, qtDone) +//line app/vmselect/prometheus/series_response.qtpl:31 qs422016 := string(qb422016.B) -//line app/vmselect/prometheus/series_response.qtpl:25 +//line app/vmselect/prometheus/series_response.qtpl:31 qt422016.ReleaseByteBuffer(qb422016) -//line app/vmselect/prometheus/series_response.qtpl:25 +//line app/vmselect/prometheus/series_response.qtpl:31 return qs422016 -//line app/vmselect/prometheus/series_response.qtpl:25 +//line app/vmselect/prometheus/series_response.qtpl:31 } diff --git a/app/vmstorage/servers/vmselect.go b/app/vmstorage/servers/vmselect.go index 65fe87ef8..28601d2ec 100644 --- a/app/vmstorage/servers/vmselect.go +++ b/app/vmstorage/servers/vmselect.go @@ -58,7 +58,7 @@ func (api *vmstorageAPI) InitSearch(qt *querytracer.Tracer, tfss []*storage.TagF return bi, nil } -func (api *vmstorageAPI) SearchMetricNames(qt *querytracer.Tracer, tfss []*storage.TagFilters, tr storage.TimeRange, maxMetrics int, deadline uint64) ([]storage.MetricName, error) { +func (api *vmstorageAPI) SearchMetricNames(qt *querytracer.Tracer, tfss []*storage.TagFilters, tr storage.TimeRange, maxMetrics int, deadline uint64) ([]string, error) { return api.s.SearchMetricNames(qt, tfss, tr, maxMetrics, deadline) } diff --git a/docs/CHANGELOG.md b/docs/CHANGELOG.md index 6b6d5ff74..e9e86fcaf 100644 --- a/docs/CHANGELOG.md +++ b/docs/CHANGELOG.md @@ -33,6 +33,7 @@ scrape_configs: ``` * FEATURE: [query tracing](https://docs.victoriametrics.com/Single-server-VictoriaMetrics.html#query-tracing): show timestamps in query traces in human-readable format (aka `RFC3339` in UTC timezone) instead of milliseconds since Unix epoch. For example, `2022-06-27T10:32:54.506Z` instead of `1656325974506`. +* FEATURE: improve performance of [/api/v1/series](https://prometheus.io/docs/prometheus/latest/querying/api/#finding-series-by-label-matchers) requests, which return big number of time series. * BUGFIX: [vmalert](https://docs.victoriametrics.com/vmalert.html): allow using `__name__` label (aka [metric name](https://prometheus.io/docs/prometheus/latest/querying/basics/#time-series-selectors)) in alerting annotations. For example `{{ $labels.__name__ }}: Too high connection number for "{{ $labels.instance }}`. * BUGFIX: limit max memory occupied by the cache, which stores parsed regular expressions. Previously too long regular expressions passed in [MetricsQL queries](https://docs.victoriametrics.com/MetricsQL.html) could result in big amounts of used memory (e.g. multiple of gigabytes). Now the max cache size for parsed regexps is limited to a a few megabytes. diff --git a/lib/storage/metric_name.go b/lib/storage/metric_name.go index 90a23f971..cbe6316ff 100644 --- a/lib/storage/metric_name.go +++ b/lib/storage/metric_name.go @@ -3,6 +3,7 @@ package storage import ( "bytes" "fmt" + "runtime" "sort" "strconv" "strings" @@ -392,6 +393,14 @@ func (mn *MetricName) Marshal(dst []byte) []byte { return dst } +// UnmarshalString unmarshals mn from s +func (mn *MetricName) UnmarshalString(s string) error { + b := bytesutil.ToUnsafeBytes(s) + err := mn.Unmarshal(b) + runtime.KeepAlive(s) + return err +} + // Unmarshal unmarshals mn from src. func (mn *MetricName) Unmarshal(src []byte) error { if len(src) < 8 { diff --git a/lib/storage/storage.go b/lib/storage/storage.go index f60aeea26..31205d7d5 100644 --- a/lib/storage/storage.go +++ b/lib/storage/storage.go @@ -1144,8 +1144,10 @@ func nextRetentionDuration(retentionMsecs int64) time.Duration { return time.Duration(deadline-t) * time.Millisecond } -// SearchMetricNames returns metric names matching the given tfss on the given tr. -func (s *Storage) SearchMetricNames(qt *querytracer.Tracer, tfss []*TagFilters, tr TimeRange, maxMetrics int, deadline uint64) ([]MetricName, error) { +// SearchMetricNames returns marshaled metric names matching the given tfss on the given tr. +// +// The marshaled metric names must be unmarshaled via MetricName.UnmarshalString(). +func (s *Storage) SearchMetricNames(qt *querytracer.Tracer, tfss []*TagFilters, tr TimeRange, maxMetrics int, deadline uint64) ([]string, error) { qt = qt.NewChild("search for matching metric names: filters=%s, timeRange=%s", tfss, &tr) defer qt.Done() tsids, err := s.searchTSIDs(qt, tfss, tr, maxMetrics, deadline) @@ -1161,7 +1163,8 @@ func (s *Storage) SearchMetricNames(qt *querytracer.Tracer, tfss []*TagFilters, accountID := tsids[0].AccountID projectID := tsids[0].ProjectID idb := s.idb() - mns := make([]MetricName, 0, len(tsids)) + metricNames := make([]string, 0, len(tsids)) + metricNamesSeen := make(map[string]struct{}, len(tsids)) var metricName []byte for i := range tsids { if i&paceLimiterSlowIterationsMask == 0 { @@ -1180,14 +1183,15 @@ func (s *Storage) SearchMetricNames(qt *querytracer.Tracer, tfss []*TagFilters, } return nil, fmt.Errorf("error when searching metricName for metricID=%d: %w", metricID, err) } - mns = mns[:len(mns)+1] - mn := &mns[len(mns)-1] - if err = mn.Unmarshal(metricName); err != nil { - return nil, fmt.Errorf("cannot unmarshal metricName=%q: %w", metricName, err) + if _, ok := metricNamesSeen[string(metricName)]; ok { + // The given metric name was already seen; skip it + continue } + metricNames = append(metricNames, string(metricName)) + metricNamesSeen[metricNames[len(metricNames)-1]] = struct{}{} } - qt.Printf("loaded %d metric names", len(mns)) - return mns, nil + qt.Printf("loaded %d metric names", len(metricNames)) + return metricNames, nil } // searchTSIDs returns sorted TSIDs for the given tfss and the given tr. diff --git a/lib/storage/storage_test.go b/lib/storage/storage_test.go index 768c39483..f49505baf 100644 --- a/lib/storage/storage_test.go +++ b/lib/storage/storage_test.go @@ -934,14 +934,21 @@ func testStorageRegisterMetricNames(s *Storage) error { if err := tfs.Add([]byte("add_id"), []byte("0"), false, false); err != nil { return fmt.Errorf("unexpected error in TagFilters.Add: %w", err) } - mns, err := s.SearchMetricNames(nil, []*TagFilters{tfs}, tr, metricsPerAdd*addsCount*100+100, noDeadline) + metricNames, err := s.SearchMetricNames(nil, []*TagFilters{tfs}, tr, metricsPerAdd*addsCount*100+100, noDeadline) if err != nil { return fmt.Errorf("error in SearchMetricNames: %w", err) } - if len(mns) < metricsPerAdd { - return fmt.Errorf("unexpected number of metricNames returned from SearchMetricNames; got %d; want at least %d", len(mns), int(metricsPerAdd)) + if err != nil { + return fmt.Errorf("cannot unmarshal metric names: %w", err) } - for i, mn := range mns { + if len(metricNames) < metricsPerAdd { + return fmt.Errorf("unexpected number of metricNames returned from SearchMetricNames; got %d; want at least %d", len(metricNames), int(metricsPerAdd)) + } + var mn MetricName + for i, metricName := range metricNames { + if err := mn.UnmarshalString(metricName); err != nil { + return fmt.Errorf("cannot unmarshal metricName=%q: %w", metricName, err) + } addID := mn.GetTagValue("add_id") if string(addID) != "0" { return fmt.Errorf("unexpected addID for metricName #%d; got %q; want %q", i, addID, "0") @@ -957,12 +964,12 @@ func testStorageRegisterMetricNames(s *Storage) error { if err := tfs.Add([]byte("add_id"), []byte("0"), false, false); err != nil { return fmt.Errorf("unexpected error in TagFilters.Add: %w", err) } - mns, err = s.SearchMetricNames(nil, []*TagFilters{tfs}, tr, metricsPerAdd*addsCount*100+100, noDeadline) + metricNames, err = s.SearchMetricNames(nil, []*TagFilters{tfs}, tr, metricsPerAdd*addsCount*100+100, noDeadline) if err != nil { return fmt.Errorf("error in SearchMetricNames for incorrect accountID, projectID: %w", err) } - if len(mns) > 0 { - return fmt.Errorf("SearchMetricNames with incorrect accountID, projectID returns unexpected non-empty result:\n%+v", mns) + if len(metricNames) > 0 { + return fmt.Errorf("SearchMetricNames with incorrect accountID, projectID returns unexpected non-empty result:\n%+v", metricNames) } return nil diff --git a/lib/vmselectapi/api.go b/lib/vmselectapi/api.go index 2d77acc4b..b4f2f3ca0 100644 --- a/lib/vmselectapi/api.go +++ b/lib/vmselectapi/api.go @@ -13,7 +13,7 @@ type API interface { InitSearch(qt *querytracer.Tracer, tfss []*storage.TagFilters, tr storage.TimeRange, maxMetrics int, deadline uint64) (BlockIterator, error) // SearchMetricNames returns metric names matching the given tfss. - SearchMetricNames(qt *querytracer.Tracer, tfss []*storage.TagFilters, tr storage.TimeRange, maxMetrics int, deadline uint64) ([]storage.MetricName, error) + SearchMetricNames(qt *querytracer.Tracer, tfss []*storage.TagFilters, tr storage.TimeRange, maxMetrics int, deadline uint64) ([]string, error) // LabelValues returns values for labelName label acorss series matching the given tfss. LabelValues(qt *querytracer.Tracer, accountID, projectID uint32, tfss []*storage.TagFilters, tr storage.TimeRange, labelName string, maxLabelValues, maxMetrics int, deadline uint64) ([]string, error) diff --git a/lib/vmselectapi/server.go b/lib/vmselectapi/server.go index 4c5d3b184..97b362b0c 100644 --- a/lib/vmselectapi/server.go +++ b/lib/vmselectapi/server.go @@ -475,7 +475,7 @@ func (s *Server) processRequest(ctx *vmselectRequestCtx) error { func (s *Server) processRPC(ctx *vmselectRequestCtx, rpcName string) error { switch rpcName { case "search_v7": - return s.processSeriesSearch(ctx) + return s.processSearch(ctx) case "searchMetricNames_v3": return s.processSearchMetricNames(ctx) case "labelValues_v5": @@ -860,7 +860,7 @@ func (s *Server) processSearchMetricNames(ctx *vmselectRequestCtx) error { if err != nil { return ctx.writeErrorMessage(err) } - mns, err := s.api.SearchMetricNames(ctx.qt, tfss, tr, maxMetrics, ctx.deadline) + metricNames, err := s.api.SearchMetricNames(ctx.qt, tfss, tr, maxMetrics, ctx.deadline) if err != nil { return ctx.writeErrorMessage(err) } @@ -871,21 +871,20 @@ func (s *Server) processSearchMetricNames(ctx *vmselectRequestCtx) error { } // Send response. - metricNamesCount := len(mns) + metricNamesCount := len(metricNames) if err := ctx.writeUint64(uint64(metricNamesCount)); err != nil { return fmt.Errorf("cannot send metricNamesCount: %w", err) } - for i, mn := range mns { - ctx.dataBuf = mn.Marshal(ctx.dataBuf[:0]) - if err := ctx.writeDataBufBytes(); err != nil { + for i, metricName := range metricNames { + if err := ctx.writeString(metricName); err != nil { return fmt.Errorf("cannot send metricName #%d: %w", i+1, err) } } - ctx.qt.Printf("sent %d series to vmselect", len(mns)) + ctx.qt.Printf("sent %d series to vmselect", len(metricNames)) return nil } -func (s *Server) processSeriesSearch(ctx *vmselectRequestCtx) error { +func (s *Server) processSearch(ctx *vmselectRequestCtx) error { s.searchRequests.Inc() // Read request.