mirror of
https://github.com/prometheus/prometheus.git
synced 2024-12-25 05:34:05 -08:00
Add fanin tests and fix uncovered bugs
This commit is contained in:
parent
9b33cfc457
commit
94acd3f1d8
|
@ -97,22 +97,25 @@ func (q querier) query(qFn func(q local.Querier) ([]local.SeriesIterator, error)
|
|||
}
|
||||
|
||||
func (q querier) MetricsForLabelMatchers(ctx context.Context, from, through model.Time, matcherSets ...metric.LabelMatchers) ([]metric.Metric, error) {
|
||||
// TODO: implement querying metrics from remote storage.
|
||||
return q.local.MetricsForLabelMatchers(ctx, from, through, matcherSets...)
|
||||
}
|
||||
|
||||
func (q querier) LastSampleForLabelMatchers(ctx context.Context, cutoff model.Time, matcherSets ...metric.LabelMatchers) (model.Vector, error) {
|
||||
// TODO: implement querying last samples from remote storage.
|
||||
return q.local.LastSampleForLabelMatchers(ctx, cutoff, matcherSets...)
|
||||
}
|
||||
|
||||
func (q querier) LabelValuesForLabelName(ctx context.Context, ln model.LabelName) (model.LabelValues, error) {
|
||||
// TODO: implement querying label values from remote storage.
|
||||
return q.local.LabelValuesForLabelName(ctx, ln)
|
||||
}
|
||||
|
||||
func (q querier) Close() error {
|
||||
for _, q := range append([]local.Querier{q.local}, q.remotes...) {
|
||||
if q.local != nil {
|
||||
if err := q.local.Close(); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
||||
for _, q := range q.remotes {
|
||||
if err := q.Close(); err != nil {
|
||||
return err
|
||||
}
|
||||
|
@ -132,22 +135,29 @@ type mergeIterator struct {
|
|||
|
||||
func (mit mergeIterator) ValueAtOrBeforeTime(t model.Time) model.SamplePair {
|
||||
latest := model.ZeroSamplePair
|
||||
for _, it := range append(mit.remote, mit.local) {
|
||||
if it == nil {
|
||||
// There might not be a local iterator for every remote series.
|
||||
continue
|
||||
}
|
||||
if mit.local != nil {
|
||||
latest = mit.local.ValueAtOrBeforeTime(t)
|
||||
}
|
||||
|
||||
v := it.ValueAtOrBeforeTime(t)
|
||||
if v.Timestamp.After(latest.Timestamp) {
|
||||
latest = v
|
||||
// We only need to look for a remote last sample if we don't have a local one
|
||||
// at all. If we have a local one, by definition we would not care about earlier
|
||||
// "last" samples, and we would not consider later ones as well, because we
|
||||
// generally only consider remote samples that are older than the oldest
|
||||
// local sample.
|
||||
if latest == model.ZeroSamplePair {
|
||||
for _, it := range mit.remote {
|
||||
v := it.ValueAtOrBeforeTime(t)
|
||||
if v.Timestamp.After(latest.Timestamp) {
|
||||
latest = v
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return latest
|
||||
}
|
||||
|
||||
func (mit mergeIterator) RangeValues(interval metric.Interval) []model.SamplePair {
|
||||
remoteCutoff := model.Earliest
|
||||
remoteCutoff := model.Latest
|
||||
var values []model.SamplePair
|
||||
if mit.local != nil {
|
||||
values = mit.local.RangeValues(interval)
|
||||
|
@ -177,7 +187,9 @@ func (mit mergeIterator) Metric() metric.Metric {
|
|||
}
|
||||
|
||||
func (mit mergeIterator) Close() {
|
||||
mit.local.Close()
|
||||
if mit.local != nil {
|
||||
mit.local.Close()
|
||||
}
|
||||
for _, it := range mit.remote {
|
||||
it.Close()
|
||||
}
|
||||
|
@ -213,6 +225,6 @@ func mergeSamples(a, b []model.SamplePair) []model.SamplePair {
|
|||
}
|
||||
}
|
||||
result = append(result, a[i:]...)
|
||||
result = append(result, b[i:]...)
|
||||
result = append(result, b[j:]...)
|
||||
return result
|
||||
}
|
||||
|
|
616
storage/fanin/fanin_test.go
Normal file
616
storage/fanin/fanin_test.go
Normal file
|
@ -0,0 +1,616 @@
|
|||
// Copyright 2017 The Prometheus Authors
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// http://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
package fanin
|
||||
|
||||
import (
|
||||
"reflect"
|
||||
"sort"
|
||||
"testing"
|
||||
"time"
|
||||
|
||||
"github.com/prometheus/common/model"
|
||||
"golang.org/x/net/context"
|
||||
|
||||
"github.com/prometheus/prometheus/storage/local"
|
||||
"github.com/prometheus/prometheus/storage/metric"
|
||||
"github.com/prometheus/prometheus/storage/remote"
|
||||
)
|
||||
|
||||
type testQuerier struct {
|
||||
series model.Matrix
|
||||
}
|
||||
|
||||
func (q testQuerier) QueryRange(ctx context.Context, from, through model.Time, matchers ...*metric.LabelMatcher) ([]local.SeriesIterator, error) {
|
||||
var outMatrix model.Matrix
|
||||
for _, s := range q.series {
|
||||
for _, m := range matchers {
|
||||
if !m.Match(s.Metric[m.Name]) {
|
||||
continue
|
||||
}
|
||||
}
|
||||
|
||||
fromIdx := sort.Search(len(s.Values), func(i int) bool {
|
||||
return !s.Values[i].Timestamp.Before(from)
|
||||
})
|
||||
throughIdx := sort.Search(len(s.Values), func(i int) bool {
|
||||
return s.Values[i].Timestamp.After(through)
|
||||
})
|
||||
|
||||
outMatrix = append(outMatrix, &model.SampleStream{
|
||||
Metric: s.Metric,
|
||||
Values: s.Values[fromIdx:throughIdx],
|
||||
})
|
||||
}
|
||||
|
||||
return remote.MatrixToIterators(outMatrix, nil)
|
||||
}
|
||||
|
||||
func (q testQuerier) QueryInstant(ctx context.Context, ts model.Time, stalenessDelta time.Duration, matchers ...*metric.LabelMatcher) ([]local.SeriesIterator, error) {
|
||||
return q.QueryRange(ctx, ts.Add(-stalenessDelta), ts, matchers...)
|
||||
}
|
||||
|
||||
func (q testQuerier) MetricsForLabelMatchers(ctx context.Context, from, through model.Time, matcherSets ...metric.LabelMatchers) ([]metric.Metric, error) {
|
||||
fpToMetric := map[model.Fingerprint]model.Metric{}
|
||||
for _, s := range q.series {
|
||||
matched := false
|
||||
for _, matchers := range matcherSets {
|
||||
for _, m := range matchers {
|
||||
if !m.Match(s.Metric[m.Name]) {
|
||||
continue
|
||||
}
|
||||
}
|
||||
|
||||
matched = true
|
||||
}
|
||||
|
||||
fromIdx := sort.Search(len(s.Values), func(i int) bool {
|
||||
return !s.Values[i].Timestamp.Before(from)
|
||||
})
|
||||
throughIdx := sort.Search(len(s.Values), func(i int) bool {
|
||||
return s.Values[i].Timestamp.After(through)
|
||||
})
|
||||
|
||||
if fromIdx == throughIdx {
|
||||
continue
|
||||
}
|
||||
|
||||
if matched {
|
||||
fpToMetric[s.Metric.Fingerprint()] = s.Metric
|
||||
}
|
||||
}
|
||||
|
||||
metrics := make([]metric.Metric, 0, len(fpToMetric))
|
||||
for _, m := range fpToMetric {
|
||||
metrics = append(metrics, metric.Metric{Metric: m})
|
||||
}
|
||||
return metrics, nil
|
||||
}
|
||||
|
||||
func (q testQuerier) LastSampleForLabelMatchers(ctx context.Context, cutoff model.Time, matcherSets ...metric.LabelMatchers) (model.Vector, error) {
|
||||
panic("not implemented")
|
||||
}
|
||||
|
||||
func (q testQuerier) LabelValuesForLabelName(ctx context.Context, ln model.LabelName) (model.LabelValues, error) {
|
||||
panic("not implemented")
|
||||
}
|
||||
|
||||
func (q testQuerier) Close() error {
|
||||
return nil
|
||||
}
|
||||
|
||||
func TestQueryRange(t *testing.T) {
|
||||
type query struct {
|
||||
from model.Time
|
||||
through model.Time
|
||||
out model.Matrix
|
||||
}
|
||||
|
||||
tests := []struct {
|
||||
name string
|
||||
local model.Matrix
|
||||
remote []model.Matrix
|
||||
queries []query
|
||||
}{
|
||||
{
|
||||
name: "duplicate samples are eliminated",
|
||||
local: model.Matrix{
|
||||
&model.SampleStream{
|
||||
Metric: model.Metric{
|
||||
model.MetricNameLabel: "testmetric",
|
||||
},
|
||||
Values: []model.SamplePair{
|
||||
{
|
||||
Timestamp: 1,
|
||||
Value: 1,
|
||||
},
|
||||
{
|
||||
Timestamp: 2,
|
||||
Value: 2,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
remote: []model.Matrix{
|
||||
model.Matrix{
|
||||
&model.SampleStream{
|
||||
Metric: model.Metric{
|
||||
model.MetricNameLabel: "testmetric",
|
||||
},
|
||||
Values: []model.SamplePair{
|
||||
{
|
||||
Timestamp: 0,
|
||||
Value: 0,
|
||||
},
|
||||
{
|
||||
Timestamp: 1,
|
||||
Value: 1,
|
||||
},
|
||||
{
|
||||
Timestamp: 2,
|
||||
Value: 2,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
model.Matrix{
|
||||
&model.SampleStream{
|
||||
Metric: model.Metric{
|
||||
model.MetricNameLabel: "testmetric",
|
||||
},
|
||||
Values: []model.SamplePair{
|
||||
{
|
||||
Timestamp: 0,
|
||||
Value: 0,
|
||||
},
|
||||
{
|
||||
Timestamp: 1,
|
||||
Value: 1,
|
||||
},
|
||||
{
|
||||
Timestamp: 2,
|
||||
Value: 2,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
queries: []query{
|
||||
{
|
||||
from: 0,
|
||||
through: 1,
|
||||
out: model.Matrix{
|
||||
&model.SampleStream{
|
||||
Metric: model.Metric{
|
||||
model.MetricNameLabel: "testmetric",
|
||||
},
|
||||
Values: []model.SamplePair{
|
||||
{
|
||||
Timestamp: 0,
|
||||
Value: 0,
|
||||
},
|
||||
{
|
||||
Timestamp: 1,
|
||||
Value: 1,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
|
||||
{
|
||||
name: "remote data is thrown away after first local sample",
|
||||
local: model.Matrix{
|
||||
&model.SampleStream{
|
||||
Metric: model.Metric{
|
||||
model.MetricNameLabel: "testmetric",
|
||||
},
|
||||
Values: []model.SamplePair{
|
||||
{
|
||||
Timestamp: 2,
|
||||
Value: 2,
|
||||
},
|
||||
{
|
||||
Timestamp: 4,
|
||||
Value: 4,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
remote: []model.Matrix{
|
||||
model.Matrix{
|
||||
&model.SampleStream{
|
||||
Metric: model.Metric{
|
||||
model.MetricNameLabel: "testmetric",
|
||||
},
|
||||
Values: []model.SamplePair{
|
||||
{
|
||||
Timestamp: 0,
|
||||
Value: 0,
|
||||
},
|
||||
{
|
||||
Timestamp: 2,
|
||||
Value: 20,
|
||||
},
|
||||
{
|
||||
Timestamp: 4,
|
||||
Value: 40,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
model.Matrix{
|
||||
&model.SampleStream{
|
||||
Metric: model.Metric{
|
||||
model.MetricNameLabel: "testmetric",
|
||||
},
|
||||
Values: []model.SamplePair{
|
||||
{
|
||||
Timestamp: 1,
|
||||
Value: 10,
|
||||
},
|
||||
{
|
||||
Timestamp: 3,
|
||||
Value: 30,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
queries: []query{
|
||||
{
|
||||
from: 0,
|
||||
through: 4,
|
||||
out: model.Matrix{
|
||||
&model.SampleStream{
|
||||
Metric: model.Metric{
|
||||
model.MetricNameLabel: "testmetric",
|
||||
},
|
||||
Values: []model.SamplePair{
|
||||
{
|
||||
Timestamp: 0,
|
||||
Value: 0,
|
||||
},
|
||||
{
|
||||
Timestamp: 1,
|
||||
Value: 10,
|
||||
},
|
||||
{
|
||||
Timestamp: 2,
|
||||
Value: 2,
|
||||
},
|
||||
{
|
||||
Timestamp: 4,
|
||||
Value: 4,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
{
|
||||
from: 2,
|
||||
through: 2,
|
||||
out: model.Matrix{
|
||||
&model.SampleStream{
|
||||
Metric: model.Metric{
|
||||
model.MetricNameLabel: "testmetric",
|
||||
},
|
||||
Values: []model.SamplePair{
|
||||
{
|
||||
Timestamp: 2,
|
||||
Value: 2,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
|
||||
{
|
||||
name: "no local data",
|
||||
remote: []model.Matrix{
|
||||
model.Matrix{
|
||||
&model.SampleStream{
|
||||
Metric: model.Metric{
|
||||
model.MetricNameLabel: "testmetric",
|
||||
},
|
||||
Values: []model.SamplePair{
|
||||
{
|
||||
Timestamp: 0,
|
||||
Value: 0,
|
||||
},
|
||||
{
|
||||
Timestamp: 2,
|
||||
Value: 20,
|
||||
},
|
||||
|
||||
{
|
||||
Timestamp: 4,
|
||||
Value: 40,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
model.Matrix{
|
||||
&model.SampleStream{
|
||||
Metric: model.Metric{
|
||||
model.MetricNameLabel: "testmetric",
|
||||
},
|
||||
Values: []model.SamplePair{
|
||||
{
|
||||
Timestamp: 1,
|
||||
Value: 10,
|
||||
},
|
||||
{
|
||||
Timestamp: 3,
|
||||
Value: 30,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
queries: []query{
|
||||
{
|
||||
from: 0,
|
||||
through: 4,
|
||||
out: model.Matrix{
|
||||
&model.SampleStream{
|
||||
Metric: model.Metric{
|
||||
model.MetricNameLabel: "testmetric",
|
||||
},
|
||||
Values: []model.SamplePair{
|
||||
{
|
||||
Timestamp: 0,
|
||||
Value: 0,
|
||||
},
|
||||
{
|
||||
Timestamp: 1,
|
||||
Value: 10,
|
||||
},
|
||||
{
|
||||
Timestamp: 2,
|
||||
Value: 20,
|
||||
},
|
||||
{
|
||||
Timestamp: 3,
|
||||
Value: 30,
|
||||
},
|
||||
{
|
||||
Timestamp: 4,
|
||||
Value: 40,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
{
|
||||
from: 2,
|
||||
through: 2,
|
||||
out: model.Matrix{
|
||||
&model.SampleStream{
|
||||
Metric: model.Metric{
|
||||
model.MetricNameLabel: "testmetric",
|
||||
},
|
||||
Values: []model.SamplePair{
|
||||
{
|
||||
Timestamp: 2,
|
||||
Value: 20,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
|
||||
{
|
||||
name: "only local data",
|
||||
local: model.Matrix{
|
||||
&model.SampleStream{
|
||||
Metric: model.Metric{
|
||||
model.MetricNameLabel: "testmetric",
|
||||
},
|
||||
Values: []model.SamplePair{
|
||||
{
|
||||
Timestamp: 0,
|
||||
Value: 0,
|
||||
},
|
||||
{
|
||||
Timestamp: 1,
|
||||
Value: 1,
|
||||
},
|
||||
|
||||
{
|
||||
Timestamp: 2,
|
||||
Value: 2,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
queries: []query{
|
||||
{
|
||||
from: 0,
|
||||
through: 4,
|
||||
out: model.Matrix{
|
||||
&model.SampleStream{
|
||||
Metric: model.Metric{
|
||||
model.MetricNameLabel: "testmetric",
|
||||
},
|
||||
Values: []model.SamplePair{
|
||||
{
|
||||
Timestamp: 0,
|
||||
Value: 0,
|
||||
},
|
||||
{
|
||||
Timestamp: 1,
|
||||
Value: 1,
|
||||
},
|
||||
{
|
||||
Timestamp: 2,
|
||||
Value: 2,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
{
|
||||
from: 3,
|
||||
through: 3,
|
||||
out: model.Matrix{
|
||||
&model.SampleStream{
|
||||
Metric: model.Metric{
|
||||
model.MetricNameLabel: "testmetric",
|
||||
},
|
||||
Values: []model.SamplePair{
|
||||
{
|
||||
Timestamp: 2,
|
||||
Value: 2,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
}
|
||||
|
||||
matcher, err := metric.NewLabelMatcher(metric.Equal, model.MetricNameLabel, "testmetric")
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
for _, test := range tests {
|
||||
q := querier{
|
||||
local: &testQuerier{test.local},
|
||||
}
|
||||
for _, m := range test.remote {
|
||||
q.remotes = append(q.remotes, &testQuerier{m})
|
||||
}
|
||||
|
||||
for i, query := range test.queries {
|
||||
var its []local.SeriesIterator
|
||||
var err error
|
||||
if query.from == query.through {
|
||||
its, err = q.QueryInstant(context.Background(), query.from, 5*time.Minute, matcher)
|
||||
} else {
|
||||
its, err = q.QueryRange(context.Background(), query.from, query.through, matcher)
|
||||
}
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
if err = q.Close(); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
out := make(model.Matrix, 0, len(query.out))
|
||||
for _, it := range its {
|
||||
var values []model.SamplePair
|
||||
if query.from == query.through {
|
||||
values = []model.SamplePair{it.ValueAtOrBeforeTime(query.from)}
|
||||
} else {
|
||||
values = it.RangeValues(metric.Interval{
|
||||
OldestInclusive: query.from,
|
||||
NewestInclusive: query.through,
|
||||
})
|
||||
}
|
||||
it.Close()
|
||||
|
||||
out = append(out, &model.SampleStream{
|
||||
Metric: it.Metric().Metric,
|
||||
Values: values,
|
||||
})
|
||||
}
|
||||
|
||||
sort.Sort(out)
|
||||
sort.Sort(query.out)
|
||||
|
||||
if !reflect.DeepEqual(out, query.out) {
|
||||
t.Fatalf("Test case %q, query %d: Unexpected query result;\n\ngot:\n\n%s\n\nwant:\n\n%s", test.name, i, out, query.out)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func TestMetricsForLabelMatchersIgnoresRemoteData(t *testing.T) {
|
||||
q := querier{
|
||||
local: &testQuerier{
|
||||
series: model.Matrix{
|
||||
&model.SampleStream{
|
||||
Metric: model.Metric{
|
||||
model.MetricNameLabel: "testmetric",
|
||||
"testlabel": "testvalue1",
|
||||
},
|
||||
Values: []model.SamplePair{{1, 1}},
|
||||
},
|
||||
&model.SampleStream{
|
||||
Metric: model.Metric{
|
||||
model.MetricNameLabel: "testmetric",
|
||||
"testlabel": "testvalue2",
|
||||
},
|
||||
Values: []model.SamplePair{{1, 1}},
|
||||
},
|
||||
},
|
||||
},
|
||||
remotes: []local.Querier{
|
||||
&testQuerier{
|
||||
series: model.Matrix{
|
||||
&model.SampleStream{
|
||||
Metric: model.Metric{
|
||||
model.MetricNameLabel: "testmetric",
|
||||
"testlabel": "testvalue2",
|
||||
},
|
||||
Values: []model.SamplePair{{1, 1}},
|
||||
},
|
||||
&model.SampleStream{
|
||||
Metric: model.Metric{
|
||||
model.MetricNameLabel: "testmetric",
|
||||
"testlabel": "testvalue3",
|
||||
},
|
||||
Values: []model.SamplePair{{1, 1}},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
}
|
||||
|
||||
matcher, err := metric.NewLabelMatcher(metric.Equal, model.MetricNameLabel, "testmetric")
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
got, err := q.MetricsForLabelMatchers(context.Background(), 0, 1, metric.LabelMatchers{matcher})
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
want := []metric.Metric{
|
||||
{
|
||||
Metric: model.Metric{
|
||||
model.MetricNameLabel: "testmetric",
|
||||
"testlabel": "testvalue1",
|
||||
},
|
||||
},
|
||||
{
|
||||
Metric: model.Metric{
|
||||
model.MetricNameLabel: "testmetric",
|
||||
"testlabel": "testvalue2",
|
||||
},
|
||||
},
|
||||
}
|
||||
|
||||
if !reflect.DeepEqual(want, got) {
|
||||
t.Fatalf("Unexpected metric returned;\n\nwant:\n\n%#v\n\ngot:\n\n%#v", want, got)
|
||||
}
|
||||
}
|
|
@ -73,14 +73,15 @@ type querier struct {
|
|||
}
|
||||
|
||||
func (q *querier) QueryRange(ctx context.Context, from, through model.Time, matchers ...*metric.LabelMatcher) ([]local.SeriesIterator, error) {
|
||||
return matrixToIterators(q.client.Read(ctx, from, through, matchers))
|
||||
return MatrixToIterators(q.client.Read(ctx, from, through, matchers))
|
||||
}
|
||||
|
||||
func (q *querier) QueryInstant(ctx context.Context, ts model.Time, stalenessDelta time.Duration, matchers ...*metric.LabelMatcher) ([]local.SeriesIterator, error) {
|
||||
return matrixToIterators(q.client.Read(ctx, ts.Add(-stalenessDelta), ts, matchers))
|
||||
return MatrixToIterators(q.client.Read(ctx, ts.Add(-stalenessDelta), ts, matchers))
|
||||
}
|
||||
|
||||
func matrixToIterators(m model.Matrix, err error) ([]local.SeriesIterator, error) {
|
||||
// MatrixToIterators returns series iterators for a given matrix.
|
||||
func MatrixToIterators(m model.Matrix, err error) ([]local.SeriesIterator, error) {
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue