prometheus/storage/metric/memory.go

491 lines
11 KiB
Go
Raw Normal View History

2013-02-08 09:03:26 -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.
package metric
import (
2013-05-21 09:12:02 -07:00
"sort"
"sync"
2013-02-08 09:03:26 -08:00
"time"
clientmodel "github.com/prometheus/client_golang/model"
"github.com/prometheus/prometheus/utility"
2013-02-08 09:03:26 -08:00
)
// Assuming sample rate of 1 / 15Hz, this allows for one hour's worth of
// storage per metric without any major reallocations.
const initialSeriesArenaSize = 4 * 60
2013-05-21 09:12:02 -07:00
2013-02-08 09:03:26 -08:00
// Models a given sample entry stored in the in-memory arena.
type value interface {
// Gets the given value.
get() clientmodel.SampleValue
2013-02-08 09:03:26 -08:00
}
// Models a single sample value. It presumes that there is either no subsequent
// value seen or that any subsequent values are of a different value.
type singletonValue clientmodel.SampleValue
2013-02-08 09:03:26 -08:00
func (v singletonValue) get() clientmodel.SampleValue {
return clientmodel.SampleValue(v)
2013-02-08 09:03:26 -08:00
}
type stream interface {
add(...*SamplePair)
clone() Values
expunge(age time.Time) Values
size() int
clear()
metric() clientmodel.Metric
getValueAtTime(t time.Time) Values
getBoundaryValues(in Interval) Values
getRangeValues(in Interval) Values
}
type arrayStream struct {
2013-05-21 09:12:02 -07:00
sync.RWMutex
m clientmodel.Metric
values Values
2013-02-08 09:03:26 -08:00
}
func (s *arrayStream) metric() clientmodel.Metric {
return s.m
}
func (s *arrayStream) add(v ...*SamplePair) {
2013-05-21 09:12:02 -07:00
s.Lock()
defer s.Unlock()
s.values = append(s.values, v...)
2013-02-08 09:03:26 -08:00
}
func (s *arrayStream) clone() Values {
2013-05-21 09:12:02 -07:00
s.RLock()
defer s.RUnlock()
clone := make(Values, len(s.values))
2013-05-21 09:12:02 -07:00
copy(clone, s.values)
2013-03-06 17:16:39 -08:00
2013-05-21 09:12:02 -07:00
return clone
}
2013-02-08 09:03:26 -08:00
func (s *arrayStream) expunge(t time.Time) Values {
s.Lock()
defer s.Unlock()
finder := func(i int) bool {
return s.values[i].Timestamp.After(t)
}
i := sort.Search(len(s.values), finder)
expunged := s.values[:i]
s.values = s.values[i:]
return expunged
}
func (s *arrayStream) getValueAtTime(t time.Time) Values {
2013-05-21 09:12:02 -07:00
s.RLock()
defer s.RUnlock()
// BUG(all): May be avenues for simplification.
2013-05-21 09:12:02 -07:00
l := len(s.values)
switch l {
case 0:
return Values{}
2013-05-21 09:12:02 -07:00
case 1:
return Values{s.values[0]}
2013-05-21 09:12:02 -07:00
default:
index := sort.Search(l, func(i int) bool {
return !s.values[i].Timestamp.Before(t)
})
if index == 0 {
return Values{s.values[0]}
2013-02-08 09:03:26 -08:00
}
2013-05-21 09:12:02 -07:00
if index == l {
return Values{s.values[l-1]}
2013-02-08 09:03:26 -08:00
}
2013-05-21 09:12:02 -07:00
if s.values[index].Timestamp.Equal(t) {
return Values{s.values[index]}
2013-03-06 17:16:39 -08:00
}
return Values{s.values[index-1], s.values[index]}
2013-02-08 09:03:26 -08:00
}
2013-05-21 09:12:02 -07:00
}
func (s *arrayStream) getBoundaryValues(in Interval) Values {
s.RLock()
defer s.RUnlock()
oldest := sort.Search(len(s.values), func(i int) bool {
return !s.values[i].Timestamp.Before(in.OldestInclusive)
})
newest := sort.Search(len(s.values), func(i int) bool {
return s.values[i].Timestamp.After(in.NewestInclusive)
})
resultRange := s.values[oldest:newest]
switch len(resultRange) {
case 0:
return Values{}
case 1:
return Values{resultRange[0]}
default:
return Values{resultRange[0], resultRange[len(resultRange)-1]}
}
2013-05-21 09:12:02 -07:00
}
func (s *arrayStream) getRangeValues(in Interval) Values {
2013-05-21 09:12:02 -07:00
s.RLock()
defer s.RUnlock()
oldest := sort.Search(len(s.values), func(i int) bool {
return !s.values[i].Timestamp.Before(in.OldestInclusive)
})
newest := sort.Search(len(s.values), func(i int) bool {
return s.values[i].Timestamp.After(in.NewestInclusive)
})
result := make(Values, newest-oldest)
2013-05-21 09:12:02 -07:00
copy(result, s.values[oldest:newest])
return result
2013-02-08 09:03:26 -08:00
}
func (s *arrayStream) size() int {
return len(s.values)
}
func (s *arrayStream) clear() {
s.values = Values{}
}
func newArrayStream(metric clientmodel.Metric) *arrayStream {
return &arrayStream{
m: metric,
values: make(Values, 0, initialSeriesArenaSize),
2013-02-08 09:03:26 -08:00
}
}
type memorySeriesStorage struct {
2013-05-21 09:12:02 -07:00
sync.RWMutex
wmCache *WatermarkCache
fingerprintToSeries map[clientmodel.Fingerprint]stream
labelPairToFingerprints map[LabelPair]clientmodel.Fingerprints
labelNameToFingerprints map[clientmodel.LabelName]clientmodel.Fingerprints
2013-02-08 09:03:26 -08:00
}
type MemorySeriesOptions struct {
// If provided, this WatermarkCache will be updated for any samples that are
// appended to the memorySeriesStorage.
WatermarkCache *WatermarkCache
}
func (s *memorySeriesStorage) AppendSamples(samples clientmodel.Samples) error {
2013-02-08 09:03:26 -08:00
for _, sample := range samples {
s.AppendSample(sample)
}
2013-05-16 07:02:07 -07:00
return nil
2013-02-08 09:03:26 -08:00
}
func (s *memorySeriesStorage) AppendSample(sample *clientmodel.Sample) error {
s.Lock()
defer s.Unlock()
fingerprint := &clientmodel.Fingerprint{}
fingerprint.LoadFromMetric(sample.Metric)
series := s.getOrCreateSeries(sample.Metric, fingerprint)
series.add(&SamplePair{
Value: sample.Value,
Timestamp: sample.Timestamp,
})
2013-02-08 09:03:26 -08:00
if s.wmCache != nil {
s.wmCache.Set(fingerprint, &watermarks{High: sample.Timestamp})
}
return nil
}
func (s *memorySeriesStorage) CreateEmptySeries(metric clientmodel.Metric) {
s.Lock()
defer s.Unlock()
fingerprint := &clientmodel.Fingerprint{}
fingerprint.LoadFromMetric(metric)
s.getOrCreateSeries(metric, fingerprint)
}
func (s *memorySeriesStorage) getOrCreateSeries(metric clientmodel.Metric, fingerprint *clientmodel.Fingerprint) stream {
series, ok := s.fingerprintToSeries[*fingerprint]
2013-02-08 09:03:26 -08:00
if !ok {
series = newArrayStream(metric)
2013-05-17 03:58:15 -07:00
s.fingerprintToSeries[*fingerprint] = series
2013-02-08 09:03:26 -08:00
for k, v := range metric {
labelPair := LabelPair{
Name: k,
Value: v,
}
2013-02-08 09:03:26 -08:00
labelPairValues := s.labelPairToFingerprints[labelPair]
labelPairValues = append(labelPairValues, fingerprint)
s.labelPairToFingerprints[labelPair] = labelPairValues
labelNameValues := s.labelNameToFingerprints[k]
labelNameValues = append(labelNameValues, fingerprint)
s.labelNameToFingerprints[k] = labelNameValues
}
}
return series
2013-02-08 09:03:26 -08:00
}
func (s *memorySeriesStorage) Flush(flushOlderThan time.Time, queue chan<- clientmodel.Samples) {
emptySeries := []clientmodel.Fingerprint{}
s.RLock()
for fingerprint, stream := range s.fingerprintToSeries {
toArchive := stream.expunge(flushOlderThan)
queued := make(clientmodel.Samples, 0, len(toArchive))
// NOTE: This duplication will go away soon.
for _, value := range toArchive {
queued = append(queued, &clientmodel.Sample{
Metric: stream.metric(),
Timestamp: value.Timestamp,
Value: value.Value,
})
}
// BUG(all): this can deadlock if the queue is full, as we only ever clear
// the queue after calling this method:
// https://github.com/prometheus/prometheus/issues/275
queue <- queued
if stream.size() == 0 {
emptySeries = append(emptySeries, fingerprint)
}
}
s.RUnlock()
s.Lock()
for _, fingerprint := range emptySeries {
if s.fingerprintToSeries[fingerprint].size() == 0 {
s.dropSeries(&fingerprint)
}
}
s.Unlock()
}
// Drop all references to a series, including any samples.
func (s *memorySeriesStorage) dropSeries(fingerprint *clientmodel.Fingerprint) {
series, ok := s.fingerprintToSeries[*fingerprint]
if !ok {
return
}
for k, v := range series.metric() {
labelPair := LabelPair{
Name: k,
Value: v,
}
delete(s.labelPairToFingerprints, labelPair)
delete(s.labelNameToFingerprints, k)
}
delete(s.fingerprintToSeries, *fingerprint)
}
// Append raw samples, bypassing indexing. Only used to add data to views,
// which don't need to lookup by metric.
func (s *memorySeriesStorage) appendSamplesWithoutIndexing(fingerprint *clientmodel.Fingerprint, samples Values) {
s.Lock()
defer s.Unlock()
series, ok := s.fingerprintToSeries[*fingerprint]
if !ok {
series = newArrayStream(clientmodel.Metric{})
s.fingerprintToSeries[*fingerprint] = series
}
series.add(samples...)
}
func (s *memorySeriesStorage) GetFingerprintsForLabelSet(l clientmodel.LabelSet) (fingerprints clientmodel.Fingerprints, err error) {
2013-05-21 09:12:02 -07:00
s.RLock()
defer s.RUnlock()
sets := []utility.Set{}
2013-02-08 09:03:26 -08:00
for k, v := range l {
values := s.labelPairToFingerprints[LabelPair{
Name: k,
Value: v,
}]
2013-02-08 09:03:26 -08:00
set := utility.Set{}
for _, fingerprint := range values {
2013-05-17 03:58:15 -07:00
set.Add(*fingerprint)
2013-02-08 09:03:26 -08:00
}
sets = append(sets, set)
}
setCount := len(sets)
if setCount == 0 {
return fingerprints, nil
2013-02-08 09:03:26 -08:00
}
base := sets[0]
for i := 1; i < setCount; i++ {
base = base.Intersection(sets[i])
}
for _, e := range base.Elements() {
fingerprint := e.(clientmodel.Fingerprint)
2013-05-17 03:58:15 -07:00
fingerprints = append(fingerprints, &fingerprint)
2013-02-08 09:03:26 -08:00
}
return fingerprints, nil
2013-02-08 09:03:26 -08:00
}
func (s *memorySeriesStorage) GetFingerprintsForLabelName(l clientmodel.LabelName) (clientmodel.Fingerprints, error) {
2013-05-21 09:12:02 -07:00
s.RLock()
2013-05-22 05:23:35 -07:00
defer s.RUnlock()
2013-05-21 09:12:02 -07:00
values, ok := s.labelNameToFingerprints[l]
if !ok {
return nil, nil
}
2013-02-08 09:03:26 -08:00
fingerprints := make(clientmodel.Fingerprints, len(values))
2013-05-21 09:12:02 -07:00
copy(fingerprints, values)
2013-02-08 09:03:26 -08:00
return fingerprints, nil
2013-02-08 09:03:26 -08:00
}
func (s *memorySeriesStorage) GetMetricForFingerprint(f *clientmodel.Fingerprint) (clientmodel.Metric, error) {
2013-05-21 09:12:02 -07:00
s.RLock()
defer s.RUnlock()
2013-05-17 03:58:15 -07:00
series, ok := s.fingerprintToSeries[*f]
2013-02-08 09:03:26 -08:00
if !ok {
return nil, nil
2013-02-08 09:03:26 -08:00
}
metric := clientmodel.Metric{}
for label, value := range series.metric() {
metric[label] = value
}
2013-02-08 09:03:26 -08:00
return metric, nil
2013-02-08 09:03:26 -08:00
}
func (s *memorySeriesStorage) HasFingerprint(f *clientmodel.Fingerprint) bool {
s.RLock()
defer s.RUnlock()
_, has := s.fingerprintToSeries[*f]
return has
}
func (s *memorySeriesStorage) CloneSamples(f *clientmodel.Fingerprint) Values {
2013-05-21 09:12:02 -07:00
s.RLock()
defer s.RUnlock()
2013-05-17 03:58:15 -07:00
series, ok := s.fingerprintToSeries[*f]
2013-02-08 09:03:26 -08:00
if !ok {
2013-05-21 09:12:02 -07:00
return nil
2013-02-08 09:03:26 -08:00
}
2013-05-21 09:12:02 -07:00
return series.clone()
}
func (s *memorySeriesStorage) GetValueAtTime(f *clientmodel.Fingerprint, t time.Time) Values {
2013-05-21 09:12:02 -07:00
s.RLock()
defer s.RUnlock()
2013-05-21 09:12:02 -07:00
series, ok := s.fingerprintToSeries[*f]
if !ok {
return nil
2013-02-08 09:03:26 -08:00
}
2013-05-21 09:12:02 -07:00
return series.getValueAtTime(t)
2013-02-08 09:03:26 -08:00
}
func (s *memorySeriesStorage) GetBoundaryValues(f *clientmodel.Fingerprint, i Interval) Values {
2013-05-21 09:12:02 -07:00
s.RLock()
defer s.RUnlock()
2013-05-17 03:58:15 -07:00
series, ok := s.fingerprintToSeries[*f]
2013-02-08 09:03:26 -08:00
if !ok {
return nil
2013-02-08 09:03:26 -08:00
}
2013-05-21 09:12:02 -07:00
return series.getBoundaryValues(i)
}
2013-02-08 09:03:26 -08:00
func (s *memorySeriesStorage) GetRangeValues(f *clientmodel.Fingerprint, i Interval) Values {
2013-05-21 09:12:02 -07:00
s.RLock()
defer s.RUnlock()
2013-05-21 09:12:02 -07:00
series, ok := s.fingerprintToSeries[*f]
2013-02-08 09:03:26 -08:00
2013-05-21 09:12:02 -07:00
if !ok {
return nil
2013-02-08 09:03:26 -08:00
}
2013-05-21 09:12:02 -07:00
return series.getRangeValues(i)
2013-02-08 09:03:26 -08:00
}
2013-05-20 11:31:58 -07:00
func (s *memorySeriesStorage) Close() {
s.Lock()
defer s.Unlock()
s.fingerprintToSeries = map[clientmodel.Fingerprint]stream{}
s.labelPairToFingerprints = map[LabelPair]clientmodel.Fingerprints{}
s.labelNameToFingerprints = map[clientmodel.LabelName]clientmodel.Fingerprints{}
2013-02-08 09:03:26 -08:00
}
func (s *memorySeriesStorage) GetAllValuesForLabel(labelName clientmodel.LabelName) (values clientmodel.LabelValues, err error) {
s.RLock()
defer s.RUnlock()
valueSet := map[clientmodel.LabelValue]bool{}
for _, series := range s.fingerprintToSeries {
if value, ok := series.metric()[labelName]; ok {
2013-03-26 06:46:02 -07:00
if !valueSet[value] {
values = append(values, value)
valueSet[value] = true
}
}
}
return
2013-02-08 09:03:26 -08:00
}
func NewMemorySeriesStorage(o MemorySeriesOptions) *memorySeriesStorage {
2013-05-20 11:31:58 -07:00
return &memorySeriesStorage{
fingerprintToSeries: make(map[clientmodel.Fingerprint]stream),
labelPairToFingerprints: make(map[LabelPair]clientmodel.Fingerprints),
labelNameToFingerprints: make(map[clientmodel.LabelName]clientmodel.Fingerprints),
wmCache: o.WatermarkCache,
2013-02-08 09:03:26 -08:00
}
}