Fix memory series indexing bug.

This fixes https://github.com/prometheus/prometheus/issues/381.

For any stale series we dropped from memory, this bug caused us to also drop
any other series from the labelpair->fingerprints memory index if they had any
label/value-pairs in common with the intentionally dropped series.

To fix this issue more easily, I converted the labelpair->fingerprints index
map values to a utility.Set of clientmodel.Fingerprints. This makes handling
this index much easier in general.

Change-Id: If5e81e202e8c542261bbd9797aa1257376c5c074
This commit is contained in:
Julius Volz 2014-03-05 23:35:49 +01:00
parent c6013ff309
commit d6827b6898
2 changed files with 87 additions and 16 deletions

View file

@ -176,7 +176,7 @@ type memorySeriesStorage struct {
wmCache *watermarkCache wmCache *watermarkCache
fingerprintToSeries map[clientmodel.Fingerprint]stream fingerprintToSeries map[clientmodel.Fingerprint]stream
labelPairToFingerprints map[LabelPair]clientmodel.Fingerprints labelPairToFingerprints map[LabelPair]utility.Set
} }
// MemorySeriesOptions bundles options used by NewMemorySeriesStorage to create // MemorySeriesOptions bundles options used by NewMemorySeriesStorage to create
@ -240,9 +240,13 @@ func (s *memorySeriesStorage) getOrCreateSeries(metric clientmodel.Metric, finge
Name: k, Name: k,
Value: v, Value: v,
} }
labelPairValues := s.labelPairToFingerprints[labelPair]
labelPairValues = append(labelPairValues, fingerprint) set, ok := s.labelPairToFingerprints[labelPair]
s.labelPairToFingerprints[labelPair] = labelPairValues if !ok {
set = utility.Set{}
s.labelPairToFingerprints[labelPair] = set
}
set.Add(*fingerprint)
} }
} }
return series return series
@ -267,7 +271,9 @@ func (s *memorySeriesStorage) Flush(flushOlderThan clientmodel.Timestamp, queue
// BUG(all): this can deadlock if the queue is full, as we only ever clear // BUG(all): this can deadlock if the queue is full, as we only ever clear
// the queue after calling this method: // the queue after calling this method:
// https://github.com/prometheus/prometheus/issues/275 // https://github.com/prometheus/prometheus/issues/275
queue <- queued if len(queued) > 0 {
queue <- queued
}
if stream.size() == 0 { if stream.size() == 0 {
emptySeries = append(emptySeries, fingerprint) emptySeries = append(emptySeries, fingerprint)
@ -280,7 +286,6 @@ func (s *memorySeriesStorage) Flush(flushOlderThan clientmodel.Timestamp, queue
s.Lock() s.Lock()
s.dropSeries(&fingerprint) s.dropSeries(&fingerprint)
s.Unlock() s.Unlock()
} }
} }
} }
@ -291,12 +296,18 @@ func (s *memorySeriesStorage) dropSeries(fingerprint *clientmodel.Fingerprint) {
if !ok { if !ok {
return return
} }
for k, v := range series.metric() { for k, v := range series.metric() {
labelPair := LabelPair{ labelPair := LabelPair{
Name: k, Name: k,
Value: v, Value: v,
} }
delete(s.labelPairToFingerprints, labelPair) if set, ok := s.labelPairToFingerprints[labelPair]; ok {
set.Remove(*fingerprint)
if len(set) == 0 {
delete(s.labelPairToFingerprints, labelPair)
}
}
} }
delete(s.fingerprintToSeries, *fingerprint) delete(s.fingerprintToSeries, *fingerprint)
} }
@ -317,32 +328,33 @@ func (s *memorySeriesStorage) appendSamplesWithoutIndexing(fingerprint *clientmo
series.add(samples...) series.add(samples...)
} }
func (s *memorySeriesStorage) GetFingerprintsForLabelSet(l clientmodel.LabelSet) (fingerprints clientmodel.Fingerprints, err error) { func (s *memorySeriesStorage) GetFingerprintsForLabelSet(l clientmodel.LabelSet) (clientmodel.Fingerprints, error) {
s.RLock() s.RLock()
defer s.RUnlock() defer s.RUnlock()
sets := []utility.Set{} sets := []utility.Set{}
for k, v := range l { for k, v := range l {
values := s.labelPairToFingerprints[LabelPair{ set, ok := s.labelPairToFingerprints[LabelPair{
Name: k, Name: k,
Value: v, Value: v,
}] }]
set := utility.Set{} if !ok {
for _, fingerprint := range values { return nil, nil
set.Add(*fingerprint)
} }
sets = append(sets, set) sets = append(sets, set)
} }
setCount := len(sets) setCount := len(sets)
if setCount == 0 { if setCount == 0 {
return fingerprints, nil return nil, nil
} }
base := sets[0] base := sets[0]
for i := 1; i < setCount; i++ { for i := 1; i < setCount; i++ {
base = base.Intersection(sets[i]) base = base.Intersection(sets[i])
} }
fingerprints := clientmodel.Fingerprints{}
for _, e := range base.Elements() { for _, e := range base.Elements() {
fingerprint := e.(clientmodel.Fingerprint) fingerprint := e.(clientmodel.Fingerprint)
fingerprints = append(fingerprints, &fingerprint) fingerprints = append(fingerprints, &fingerprint)
@ -430,8 +442,8 @@ func (s *memorySeriesStorage) Close() {
s.Lock() s.Lock()
defer s.Unlock() defer s.Unlock()
s.fingerprintToSeries = map[clientmodel.Fingerprint]stream{} s.fingerprintToSeries = nil
s.labelPairToFingerprints = map[LabelPair]clientmodel.Fingerprints{} s.labelPairToFingerprints = nil
} }
func (s *memorySeriesStorage) GetAllValuesForLabel(labelName clientmodel.LabelName) (values clientmodel.LabelValues, err error) { func (s *memorySeriesStorage) GetAllValuesForLabel(labelName clientmodel.LabelName) (values clientmodel.LabelValues, err error) {
@ -455,7 +467,7 @@ func (s *memorySeriesStorage) GetAllValuesForLabel(labelName clientmodel.LabelNa
func NewMemorySeriesStorage(o MemorySeriesOptions) *memorySeriesStorage { func NewMemorySeriesStorage(o MemorySeriesOptions) *memorySeriesStorage {
return &memorySeriesStorage{ return &memorySeriesStorage{
fingerprintToSeries: make(map[clientmodel.Fingerprint]stream), fingerprintToSeries: make(map[clientmodel.Fingerprint]stream),
labelPairToFingerprints: make(map[LabelPair]clientmodel.Fingerprints), labelPairToFingerprints: make(map[LabelPair]utility.Set),
wmCache: o.WatermarkCache, wmCache: o.WatermarkCache,
} }
} }

View file

@ -93,3 +93,62 @@ func BenchmarkAppendSample100(b *testing.B) {
func BenchmarkAppendSample1000(b *testing.B) { func BenchmarkAppendSample1000(b *testing.B) {
benchmarkAppendSamples(b, 1000) benchmarkAppendSamples(b, 1000)
} }
// Regression test for https://github.com/prometheus/prometheus/issues/381.
//
// 1. Creates samples for two timeseries with one common labelpair.
// 2. Flushes memory storage such that only one series is dropped from memory.
// 3. Gets fingerprints for common labelpair.
// 4. Checks that exactly one fingerprint remains.
func TestDroppedSeriesIndexRegression(t *testing.T) {
samples := clientmodel.Samples{
&clientmodel.Sample{
Metric: clientmodel.Metric{
clientmodel.MetricNameLabel: "testmetric",
"different": "differentvalue1",
"common": "samevalue",
},
Value: 1,
Timestamp: clientmodel.TimestampFromTime(time.Date(2000, 0, 0, 0, 0, 0, 0, time.UTC)),
},
&clientmodel.Sample{
Metric: clientmodel.Metric{
clientmodel.MetricNameLabel: "testmetric",
"different": "differentvalue2",
"common": "samevalue",
},
Value: 2,
Timestamp: clientmodel.TimestampFromTime(time.Date(2002, 0, 0, 0, 0, 0, 0, time.UTC)),
},
}
s := NewMemorySeriesStorage(MemorySeriesOptions{})
s.AppendSamples(samples)
common := clientmodel.LabelSet{"common": "samevalue"}
fps, err := s.GetFingerprintsForLabelSet(common)
if err != nil {
t.Fatal(err)
}
if len(fps) != 2 {
t.Fatalf("Got %d fingerprints, expected 2", len(fps))
}
toDisk := make(chan clientmodel.Samples, 2)
s.Flush(clientmodel.TimestampFromTime(time.Date(2001, 0, 0, 0, 0, 0, 0, time.UTC)), toDisk)
if len(toDisk) != 1 {
t.Fatalf("Got %d disk sample lists, expected 1", len(toDisk))
}
diskSamples := <-toDisk
if len(diskSamples) != 1 {
t.Fatalf("Got %d disk samples, expected 1", len(diskSamples))
}
fps, err = s.GetFingerprintsForLabelSet(common)
if err != nil {
t.Fatal(err)
}
if len(fps) != 1 {
t.Fatalf("Got %d fingerprints, expected 1", len(fps))
}
}