Skip to content

Commit

Permalink
Add reason why the key was evicted in the `cortex_ingester_expanded_p…
Browse files Browse the repository at this point in the history
…ostings_cache_evicts` metric (cortexproject#6318)

* Enhancing  metric

Signed-off-by: alanprot <[email protected]>

* lint

Signed-off-by: alanprot <[email protected]>

---------

Signed-off-by: alanprot <[email protected]>
  • Loading branch information
alanprot authored Nov 7, 2024
1 parent 20686ac commit d519e78
Show file tree
Hide file tree
Showing 2 changed files with 53 additions and 12 deletions.
27 changes: 16 additions & 11 deletions pkg/storage/tsdb/expanded_postings_cache.go
Original file line number Diff line number Diff line change
Expand Up @@ -55,7 +55,7 @@ func NewPostingCacheMetrics(r prometheus.Registerer) *ExpandedPostingsCacheMetri
CacheEvicts: promauto.With(r).NewCounterVec(prometheus.CounterOpts{
Name: "cortex_ingester_expanded_postings_cache_evicts",
Help: "Total number of evictions in the cache, excluding items that got evicted due to TTL.",
}, []string{"cache"}),
}, []string{"cache", "reason"}),
NonCacheableQueries: promauto.With(r).NewCounterVec(prometheus.CounterOpts{
Name: "cortex_ingester_expanded_postings_non_cacheable_queries",
Help: "Total number of non cacheable queries.",
Expand Down Expand Up @@ -301,14 +301,15 @@ func (c *fifoCache[V]) expire() {
return
}
c.cachedMtx.RLock()
if !c.shouldEvictHead() {
if _, r := c.shouldEvictHead(); !r {
c.cachedMtx.RUnlock()
return
}
c.cachedMtx.RUnlock()
c.cachedMtx.Lock()
defer c.cachedMtx.Unlock()
for c.shouldEvictHead() {
for reason, r := c.shouldEvictHead(); r; reason, r = c.shouldEvictHead() {
c.metrics.CacheEvicts.WithLabelValues(c.name, reason).Inc()
c.evictHead()
}
}
Expand Down Expand Up @@ -340,6 +341,7 @@ func (c *fifoCache[V]) getPromiseForKey(k string, fetch func() (V, int64, error)

// If is cached but is expired, lets try to replace the cache value.
if loaded.(*cacheEntryPromise[V]).isExpired(c.cfg.Ttl, c.timeNow()) && c.cachedValues.CompareAndSwap(k, loaded, r) {
c.metrics.CacheEvicts.WithLabelValues(c.name, "expired").Inc()
r.v, r.sizeBytes, r.err = fetch()
r.sizeBytes += int64(len(k))
c.updateSize(loaded.(*cacheEntryPromise[V]).sizeBytes, r.sizeBytes)
Expand All @@ -357,22 +359,25 @@ func (c *fifoCache[V]) contains(k string) bool {
return ok
}

func (c *fifoCache[V]) shouldEvictHead() bool {
if c.cachedBytes > c.cfg.MaxBytes {
c.metrics.CacheEvicts.WithLabelValues(c.name).Inc()
return true
}
func (c *fifoCache[V]) shouldEvictHead() (string, bool) {
h := c.cached.Front()
if h == nil {
return false
return "", false
}

if c.cachedBytes > c.cfg.MaxBytes {
return "full", true
}

key := h.Value.(string)

if l, ok := c.cachedValues.Load(key); ok {
return l.(*cacheEntryPromise[V]).isExpired(c.cfg.Ttl, c.timeNow())
if l.(*cacheEntryPromise[V]).isExpired(c.cfg.Ttl, c.timeNow()) {
return "expired", true
}
}

return false
return "", false
}

func (c *fifoCache[V]) evictHead() {
Expand Down
38 changes: 37 additions & 1 deletion pkg/storage/tsdb/expanded_postings_cache_test.go
Original file line number Diff line number Diff line change
@@ -1,13 +1,15 @@
package tsdb

import (
"bytes"
"fmt"
"strings"
"sync"
"testing"
"time"

"github.com/prometheus/client_golang/prometheus"
"github.com/prometheus/client_golang/prometheus/testutil"
"github.com/stretchr/testify/require"
"go.uber.org/atomic"
)
Expand Down Expand Up @@ -88,7 +90,8 @@ func TestFifoCacheExpire(t *testing.T) {

for name, c := range tc {
t.Run(name, func(t *testing.T) {
m := NewPostingCacheMetrics(prometheus.NewPedanticRegistry())
r := prometheus.NewPedanticRegistry()
m := NewPostingCacheMetrics(r)
timeNow := time.Now
cache := newFifoCache[int](c.cfg, "test", m, timeNow)

Expand Down Expand Up @@ -118,6 +121,16 @@ func TestFifoCacheExpire(t *testing.T) {

require.Equal(t, c.expectedFinalItems, totalCacheSize)

if c.expectedFinalItems != numberOfKeys {
err := testutil.GatherAndCompare(r, bytes.NewBufferString(fmt.Sprintf(`
# HELP cortex_ingester_expanded_postings_cache_evicts Total number of evictions in the cache, excluding items that got evicted due to TTL.
# TYPE cortex_ingester_expanded_postings_cache_evicts counter
cortex_ingester_expanded_postings_cache_evicts{cache="test",reason="full"} %v
`, numberOfKeys-c.expectedFinalItems)), "cortex_ingester_expanded_postings_cache_evicts")
require.NoError(t, err)

}

if c.ttlExpire {
cache.timeNow = func() time.Time {
return timeNow().Add(2 * c.cfg.Ttl)
Expand All @@ -135,6 +148,29 @@ func TestFifoCacheExpire(t *testing.T) {
// Total Size Updated
require.Equal(t, originalSize+10, cache.cachedBytes)
}

err := testutil.GatherAndCompare(r, bytes.NewBufferString(fmt.Sprintf(`
# HELP cortex_ingester_expanded_postings_cache_evicts Total number of evictions in the cache, excluding items that got evicted due to TTL.
# TYPE cortex_ingester_expanded_postings_cache_evicts counter
cortex_ingester_expanded_postings_cache_evicts{cache="test",reason="expired"} %v
`, numberOfKeys)), "cortex_ingester_expanded_postings_cache_evicts")
require.NoError(t, err)

cache.timeNow = func() time.Time {
return timeNow().Add(5 * c.cfg.Ttl)
}

cache.getPromiseForKey("newKwy", func() (int, int64, error) {
return 2, 18, nil
})

// Should expire all keys again as ttl is expired
err = testutil.GatherAndCompare(r, bytes.NewBufferString(fmt.Sprintf(`
# HELP cortex_ingester_expanded_postings_cache_evicts Total number of evictions in the cache, excluding items that got evicted due to TTL.
# TYPE cortex_ingester_expanded_postings_cache_evicts counter
cortex_ingester_expanded_postings_cache_evicts{cache="test",reason="expired"} %v
`, numberOfKeys*2)), "cortex_ingester_expanded_postings_cache_evicts")
require.NoError(t, err)
}
})
}
Expand Down

0 comments on commit d519e78

Please sign in to comment.