mirror of
https://github.com/prometheus/prometheus.git
synced 2025-01-13 06:47:28 -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) {
|
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...)
|
return q.local.MetricsForLabelMatchers(ctx, from, through, matcherSets...)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (q querier) LastSampleForLabelMatchers(ctx context.Context, cutoff model.Time, matcherSets ...metric.LabelMatchers) (model.Vector, error) {
|
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...)
|
return q.local.LastSampleForLabelMatchers(ctx, cutoff, matcherSets...)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (q querier) LabelValuesForLabelName(ctx context.Context, ln model.LabelName) (model.LabelValues, error) {
|
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)
|
return q.local.LabelValuesForLabelName(ctx, ln)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (q querier) Close() error {
|
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 {
|
if err := q.Close(); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -132,22 +135,29 @@ type mergeIterator struct {
|
||||||
|
|
||||||
func (mit mergeIterator) ValueAtOrBeforeTime(t model.Time) model.SamplePair {
|
func (mit mergeIterator) ValueAtOrBeforeTime(t model.Time) model.SamplePair {
|
||||||
latest := model.ZeroSamplePair
|
latest := model.ZeroSamplePair
|
||||||
for _, it := range append(mit.remote, mit.local) {
|
if mit.local != nil {
|
||||||
if it == nil {
|
latest = mit.local.ValueAtOrBeforeTime(t)
|
||||||
// There might not be a local iterator for every remote series.
|
|
||||||
continue
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 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)
|
v := it.ValueAtOrBeforeTime(t)
|
||||||
if v.Timestamp.After(latest.Timestamp) {
|
if v.Timestamp.After(latest.Timestamp) {
|
||||||
latest = v
|
latest = v
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return latest
|
return latest
|
||||||
}
|
}
|
||||||
|
|
||||||
func (mit mergeIterator) RangeValues(interval metric.Interval) []model.SamplePair {
|
func (mit mergeIterator) RangeValues(interval metric.Interval) []model.SamplePair {
|
||||||
remoteCutoff := model.Earliest
|
remoteCutoff := model.Latest
|
||||||
var values []model.SamplePair
|
var values []model.SamplePair
|
||||||
if mit.local != nil {
|
if mit.local != nil {
|
||||||
values = mit.local.RangeValues(interval)
|
values = mit.local.RangeValues(interval)
|
||||||
|
@ -177,7 +187,9 @@ func (mit mergeIterator) Metric() metric.Metric {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (mit mergeIterator) Close() {
|
func (mit mergeIterator) Close() {
|
||||||
|
if mit.local != nil {
|
||||||
mit.local.Close()
|
mit.local.Close()
|
||||||
|
}
|
||||||
for _, it := range mit.remote {
|
for _, it := range mit.remote {
|
||||||
it.Close()
|
it.Close()
|
||||||
}
|
}
|
||||||
|
@ -213,6 +225,6 @@ func mergeSamples(a, b []model.SamplePair) []model.SamplePair {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
result = append(result, a[i:]...)
|
result = append(result, a[i:]...)
|
||||||
result = append(result, b[i:]...)
|
result = append(result, b[j:]...)
|
||||||
return result
|
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) {
|
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) {
|
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 {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue