2013-02-07 02:49:04 -08:00
|
|
|
// Copyright 2013 Prometheus Team
|
|
|
|
// 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.
|
|
|
|
|
2013-01-07 14:24:26 -08:00
|
|
|
package ast
|
|
|
|
|
|
|
|
import (
|
2013-01-25 03:21:44 -08:00
|
|
|
"flag"
|
2013-01-27 09:49:45 -08:00
|
|
|
"github.com/prometheus/prometheus/model"
|
|
|
|
"github.com/prometheus/prometheus/storage/metric"
|
2013-01-07 14:24:26 -08:00
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
2013-01-25 03:21:44 -08:00
|
|
|
var defaultStalenessDelta = flag.Int("defaultStalenessDelta", 300, "Default staleness delta allowance in seconds during expression evaluations.")
|
2013-01-07 14:24:26 -08:00
|
|
|
|
2013-04-18 07:10:52 -07:00
|
|
|
// Describes the lenience limits to apply to values from the materialized view.
|
|
|
|
type StalenessPolicy struct {
|
|
|
|
// Describes the inclusive limit at which individual points if requested will
|
|
|
|
// be matched and subject to interpolation.
|
|
|
|
DeltaAllowance time.Duration
|
|
|
|
}
|
|
|
|
|
2013-03-21 10:06:15 -07:00
|
|
|
type viewAdapter struct {
|
2013-05-07 04:15:10 -07:00
|
|
|
// Policy that dictates when sample values around an evaluation time are to
|
|
|
|
// be interpreted as stale.
|
2013-04-18 07:10:52 -07:00
|
|
|
stalenessPolicy StalenessPolicy
|
2013-05-07 04:15:10 -07:00
|
|
|
// AST-global storage to use for operations that are not supported by views
|
|
|
|
// (i.e. fingerprint->metric lookups).
|
|
|
|
storage *metric.TieredStorage
|
|
|
|
// The materialized view which contains all timeseries data required for
|
|
|
|
// executing a query.
|
|
|
|
view metric.View
|
2013-01-07 14:24:26 -08:00
|
|
|
}
|
|
|
|
|
2013-03-28 08:41:51 -07:00
|
|
|
// interpolateSamples interpolates a value at a target time between two
|
|
|
|
// provided sample pairs.
|
|
|
|
func interpolateSamples(first, second *model.SamplePair, timestamp time.Time) *model.SamplePair {
|
|
|
|
dv := second.Value - first.Value
|
|
|
|
dt := second.Timestamp.Sub(first.Timestamp)
|
|
|
|
|
|
|
|
dDt := dv / model.SampleValue(dt)
|
|
|
|
offset := model.SampleValue(timestamp.Sub(first.Timestamp))
|
|
|
|
|
|
|
|
return &model.SamplePair{
|
|
|
|
Value: first.Value + (offset * dDt),
|
|
|
|
Timestamp: timestamp,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// chooseClosestSample chooses the closest sample of a list of samples
|
|
|
|
// surrounding a given target time. If samples are found both before and after
|
|
|
|
// the target time, the sample value is interpolated between these. Otherwise,
|
|
|
|
// the single closest sample is returned verbatim.
|
2013-04-18 16:00:57 -07:00
|
|
|
func (v *viewAdapter) chooseClosestSample(samples model.Values, timestamp time.Time) (sample *model.SamplePair) {
|
2013-03-28 08:41:51 -07:00
|
|
|
var closestBefore *model.SamplePair
|
|
|
|
var closestAfter *model.SamplePair
|
2013-03-21 10:06:15 -07:00
|
|
|
for _, candidate := range samples {
|
2013-03-28 09:05:06 -07:00
|
|
|
delta := candidate.Timestamp.Sub(timestamp)
|
2013-03-28 08:41:51 -07:00
|
|
|
// Samples before target time.
|
2013-03-21 10:06:15 -07:00
|
|
|
if delta < 0 {
|
2013-03-28 08:41:51 -07:00
|
|
|
// Ignore samples outside of staleness policy window.
|
|
|
|
if -delta > v.stalenessPolicy.DeltaAllowance {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
// Ignore samples that are farther away than what we've seen before.
|
|
|
|
if closestBefore != nil && candidate.Timestamp.Before(closestBefore.Timestamp) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
sample := candidate
|
|
|
|
closestBefore = &sample
|
2013-03-21 10:06:15 -07:00
|
|
|
}
|
2013-01-07 14:24:26 -08:00
|
|
|
|
2013-03-28 08:41:51 -07:00
|
|
|
// Samples after target time.
|
|
|
|
if delta >= 0 {
|
|
|
|
// Ignore samples outside of staleness policy window.
|
|
|
|
if delta > v.stalenessPolicy.DeltaAllowance {
|
2013-03-21 10:06:15 -07:00
|
|
|
continue
|
|
|
|
}
|
2013-03-28 08:41:51 -07:00
|
|
|
// Ignore samples that are farther away than samples we've seen before.
|
|
|
|
if closestAfter != nil && candidate.Timestamp.After(closestAfter.Timestamp) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
sample := candidate
|
|
|
|
closestAfter = &sample
|
2013-03-21 10:06:15 -07:00
|
|
|
}
|
2013-03-28 08:41:51 -07:00
|
|
|
}
|
2013-03-21 10:06:15 -07:00
|
|
|
|
2013-03-28 08:41:51 -07:00
|
|
|
switch {
|
|
|
|
case closestBefore != nil && closestAfter != nil:
|
2013-03-28 09:05:06 -07:00
|
|
|
sample = interpolateSamples(closestBefore, closestAfter, timestamp)
|
2013-03-28 08:41:51 -07:00
|
|
|
case closestBefore != nil:
|
|
|
|
sample = closestBefore
|
|
|
|
default:
|
|
|
|
sample = closestAfter
|
2013-03-21 10:06:15 -07:00
|
|
|
}
|
2013-03-28 08:41:51 -07:00
|
|
|
|
2013-03-21 10:06:15 -07:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2013-03-28 09:05:06 -07:00
|
|
|
func (v *viewAdapter) GetValueAtTime(fingerprints model.Fingerprints, timestamp time.Time) (samples Vector, err error) {
|
2013-01-07 14:24:26 -08:00
|
|
|
for _, fingerprint := range fingerprints {
|
2013-03-28 09:05:06 -07:00
|
|
|
sampleCandidates := v.view.GetValueAtTime(fingerprint, timestamp)
|
2013-03-21 10:06:15 -07:00
|
|
|
samplePair := v.chooseClosestSample(sampleCandidates, timestamp)
|
2013-05-07 04:15:10 -07:00
|
|
|
m, err := v.storage.GetMetricForFingerprint(fingerprint)
|
2013-01-07 14:24:26 -08:00
|
|
|
if err != nil {
|
|
|
|
continue
|
|
|
|
}
|
2013-03-21 10:06:15 -07:00
|
|
|
if samplePair != nil {
|
2013-03-28 09:05:06 -07:00
|
|
|
samples = append(samples, model.Sample{
|
2013-05-14 07:25:06 -07:00
|
|
|
Metric: m,
|
2013-03-21 10:06:15 -07:00
|
|
|
Value: samplePair.Value,
|
2013-03-28 09:05:06 -07:00
|
|
|
Timestamp: timestamp,
|
2013-03-21 10:06:15 -07:00
|
|
|
})
|
|
|
|
}
|
2013-01-07 14:24:26 -08:00
|
|
|
}
|
2013-03-18 09:12:51 -07:00
|
|
|
return
|
2013-01-07 14:24:26 -08:00
|
|
|
}
|
|
|
|
|
2013-03-28 09:05:06 -07:00
|
|
|
func (v *viewAdapter) GetBoundaryValues(fingerprints model.Fingerprints, interval *model.Interval) (sampleSets []model.SampleSet, err error) {
|
2013-03-18 09:12:51 -07:00
|
|
|
for _, fingerprint := range fingerprints {
|
2013-01-12 12:22:59 -08:00
|
|
|
// TODO: change to GetBoundaryValues() once it has the right return type.
|
2013-03-21 10:06:15 -07:00
|
|
|
samplePairs := v.view.GetRangeValues(fingerprint, *interval)
|
|
|
|
if samplePairs == nil {
|
|
|
|
continue
|
2013-01-10 16:19:27 -08:00
|
|
|
}
|
2013-03-21 10:06:15 -07:00
|
|
|
|
|
|
|
// TODO: memoize/cache this.
|
2013-05-07 04:15:10 -07:00
|
|
|
m, err := v.storage.GetMetricForFingerprint(fingerprint)
|
2013-03-21 10:06:15 -07:00
|
|
|
if err != nil {
|
2013-01-10 16:19:27 -08:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2013-03-28 09:05:06 -07:00
|
|
|
sampleSet := model.SampleSet{
|
2013-05-14 07:25:06 -07:00
|
|
|
Metric: m,
|
2013-03-21 10:06:15 -07:00
|
|
|
Values: samplePairs,
|
|
|
|
}
|
2013-01-10 16:19:27 -08:00
|
|
|
sampleSets = append(sampleSets, sampleSet)
|
|
|
|
}
|
|
|
|
return sampleSets, nil
|
2013-01-07 14:24:26 -08:00
|
|
|
}
|
|
|
|
|
2013-03-28 09:05:06 -07:00
|
|
|
func (v *viewAdapter) GetRangeValues(fingerprints model.Fingerprints, interval *model.Interval) (sampleSets []model.SampleSet, err error) {
|
2013-03-18 09:12:51 -07:00
|
|
|
for _, fingerprint := range fingerprints {
|
2013-03-21 10:06:15 -07:00
|
|
|
samplePairs := v.view.GetRangeValues(fingerprint, *interval)
|
|
|
|
if samplePairs == nil {
|
|
|
|
continue
|
2013-01-10 16:19:27 -08:00
|
|
|
}
|
2013-03-21 10:06:15 -07:00
|
|
|
|
|
|
|
// TODO: memoize/cache this.
|
2013-05-07 04:15:10 -07:00
|
|
|
m, err := v.storage.GetMetricForFingerprint(fingerprint)
|
2013-03-21 10:06:15 -07:00
|
|
|
if err != nil {
|
2013-01-10 16:19:27 -08:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2013-03-28 09:05:06 -07:00
|
|
|
sampleSet := model.SampleSet{
|
2013-05-14 07:25:06 -07:00
|
|
|
Metric: m,
|
2013-03-21 10:06:15 -07:00
|
|
|
Values: samplePairs,
|
|
|
|
}
|
2013-01-10 16:19:27 -08:00
|
|
|
sampleSets = append(sampleSets, sampleSet)
|
|
|
|
}
|
|
|
|
return sampleSets, nil
|
2013-01-07 14:24:26 -08:00
|
|
|
}
|
|
|
|
|
2013-05-07 04:15:10 -07:00
|
|
|
func NewViewAdapter(view metric.View, storage *metric.TieredStorage) *viewAdapter {
|
2013-04-18 07:10:52 -07:00
|
|
|
stalenessPolicy := StalenessPolicy{
|
2013-03-21 10:06:15 -07:00
|
|
|
DeltaAllowance: time.Duration(*defaultStalenessDelta) * time.Second,
|
2013-01-25 03:21:44 -08:00
|
|
|
}
|
2013-03-21 10:06:15 -07:00
|
|
|
|
|
|
|
return &viewAdapter{
|
2013-04-18 07:10:52 -07:00
|
|
|
stalenessPolicy: stalenessPolicy,
|
2013-05-07 04:15:10 -07:00
|
|
|
storage: storage,
|
|
|
|
view: view,
|
2013-01-07 14:24:26 -08:00
|
|
|
}
|
|
|
|
}
|