mirror of
https://github.com/prometheus/prometheus.git
synced 2025-01-11 13:57:36 -08:00
Make MemPostings nested.
This saves memory, about a quarter of the size of the postings map itself with high-cardinality labels (not including the post ids). Signed-off-by: Brian Brazil <brian.brazil@robustperception.io>
This commit is contained in:
parent
fc99b8bb3a
commit
407e12d051
|
@ -320,10 +320,12 @@ func TestPersistence_index_e2e(t *testing.T) {
|
|||
testutil.Ok(t, err)
|
||||
mi.WritePostings("", "", newListPostings(all))
|
||||
|
||||
for l := range postings.m {
|
||||
err = iw.WritePostings(l.Name, l.Value, postings.Get(l.Name, l.Value))
|
||||
for n, e := range postings.m {
|
||||
for v := range e {
|
||||
err = iw.WritePostings(n, v, postings.Get(n, v))
|
||||
testutil.Ok(t, err)
|
||||
mi.WritePostings(l.Name, l.Value, postings.Get(l.Name, l.Value))
|
||||
mi.WritePostings(n, v, postings.Get(n, v))
|
||||
}
|
||||
}
|
||||
|
||||
err = iw.Close()
|
||||
|
|
|
@ -36,14 +36,14 @@ func AllPostingsKey() (name, value string) {
|
|||
// unordered batch fills on startup.
|
||||
type MemPostings struct {
|
||||
mtx sync.RWMutex
|
||||
m map[labels.Label][]uint64
|
||||
m map[string]map[string][]uint64
|
||||
ordered bool
|
||||
}
|
||||
|
||||
// NewMemPostings returns a memPostings that's ready for reads and writes.
|
||||
func NewMemPostings() *MemPostings {
|
||||
return &MemPostings{
|
||||
m: make(map[labels.Label][]uint64, 512),
|
||||
m: make(map[string]map[string][]uint64, 512),
|
||||
ordered: true,
|
||||
}
|
||||
}
|
||||
|
@ -52,7 +52,7 @@ func NewMemPostings() *MemPostings {
|
|||
// until ensureOrder was called once.
|
||||
func NewUnorderedMemPostings() *MemPostings {
|
||||
return &MemPostings{
|
||||
m: make(map[labels.Label][]uint64, 512),
|
||||
m: make(map[string]map[string][]uint64, 512),
|
||||
ordered: false,
|
||||
}
|
||||
}
|
||||
|
@ -62,8 +62,10 @@ func (p *MemPostings) SortedKeys() []labels.Label {
|
|||
p.mtx.RLock()
|
||||
keys := make([]labels.Label, 0, len(p.m))
|
||||
|
||||
for l := range p.m {
|
||||
keys = append(keys, l)
|
||||
for n, e := range p.m {
|
||||
for v := range e {
|
||||
keys = append(keys, labels.Label{Name: n, Value: v})
|
||||
}
|
||||
}
|
||||
p.mtx.RUnlock()
|
||||
|
||||
|
@ -78,14 +80,18 @@ func (p *MemPostings) SortedKeys() []labels.Label {
|
|||
|
||||
// Get returns a postings list for the given label pair.
|
||||
func (p *MemPostings) Get(name, value string) Postings {
|
||||
var lp []uint64
|
||||
p.mtx.RLock()
|
||||
l := p.m[labels.Label{Name: name, Value: value}]
|
||||
l := p.m[name]
|
||||
if l != nil {
|
||||
lp = l[value]
|
||||
}
|
||||
p.mtx.RUnlock()
|
||||
|
||||
if l == nil {
|
||||
if lp == nil {
|
||||
return EmptyPostings()
|
||||
}
|
||||
return newListPostings(l)
|
||||
return newListPostings(lp)
|
||||
}
|
||||
|
||||
// All returns a postings list over all documents ever added.
|
||||
|
@ -118,9 +124,11 @@ func (p *MemPostings) EnsureOrder() {
|
|||
}()
|
||||
}
|
||||
|
||||
for _, l := range p.m {
|
||||
for _, e := range p.m {
|
||||
for _, l := range e {
|
||||
workc <- l
|
||||
}
|
||||
}
|
||||
close(workc)
|
||||
wg.Wait()
|
||||
|
||||
|
@ -129,24 +137,32 @@ func (p *MemPostings) EnsureOrder() {
|
|||
|
||||
// Delete removes all ids in the given map from the postings lists.
|
||||
func (p *MemPostings) Delete(deleted map[uint64]struct{}) {
|
||||
var keys []labels.Label
|
||||
var keys, vals []string
|
||||
|
||||
// Collect all keys relevant for deletion once. New keys added afterwards
|
||||
// can by definition not be affected by any of the given deletes.
|
||||
p.mtx.RLock()
|
||||
for l := range p.m {
|
||||
keys = append(keys, l)
|
||||
for n := range p.m {
|
||||
keys = append(keys, n)
|
||||
}
|
||||
p.mtx.RUnlock()
|
||||
|
||||
// For each key we first analyse whether the postings list is affected by the deletes.
|
||||
for _, n := range keys {
|
||||
p.mtx.RLock()
|
||||
vals = vals[:0]
|
||||
for v := range p.m[n] {
|
||||
vals = append(vals, v)
|
||||
}
|
||||
p.mtx.RUnlock()
|
||||
|
||||
// For each posting we first analyse whether the postings list is affected by the deletes.
|
||||
// If yes, we actually reallocate a new postings list.
|
||||
for _, l := range keys {
|
||||
for _, l := range vals {
|
||||
// Only lock for processing one postings list so we don't block reads for too long.
|
||||
p.mtx.Lock()
|
||||
|
||||
found := false
|
||||
for _, id := range p.m[l] {
|
||||
for _, id := range p.m[n][l] {
|
||||
if _, ok := deleted[id]; ok {
|
||||
found = true
|
||||
break
|
||||
|
@ -156,17 +172,23 @@ func (p *MemPostings) Delete(deleted map[uint64]struct{}) {
|
|||
p.mtx.Unlock()
|
||||
continue
|
||||
}
|
||||
repl := make([]uint64, 0, len(p.m[l]))
|
||||
repl := make([]uint64, 0, len(p.m[n][l]))
|
||||
|
||||
for _, id := range p.m[l] {
|
||||
for _, id := range p.m[n][l] {
|
||||
if _, ok := deleted[id]; !ok {
|
||||
repl = append(repl, id)
|
||||
}
|
||||
}
|
||||
if len(repl) > 0 {
|
||||
p.m[l] = repl
|
||||
p.m[n][l] = repl
|
||||
} else {
|
||||
delete(p.m, l)
|
||||
delete(p.m[n], l)
|
||||
}
|
||||
p.mtx.Unlock()
|
||||
}
|
||||
p.mtx.Lock()
|
||||
if len(p.m[n]) == 0 {
|
||||
delete(p.m, n)
|
||||
}
|
||||
p.mtx.Unlock()
|
||||
}
|
||||
|
@ -177,11 +199,13 @@ func (p *MemPostings) Iter(f func(labels.Label, Postings) error) error {
|
|||
p.mtx.RLock()
|
||||
defer p.mtx.RUnlock()
|
||||
|
||||
for l, p := range p.m {
|
||||
if err := f(l, newListPostings(p)); err != nil {
|
||||
for n, e := range p.m {
|
||||
for v, p := range e {
|
||||
if err := f(labels.Label{Name: n, Value: v}, newListPostings(p)); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
|
@ -198,8 +222,13 @@ func (p *MemPostings) Add(id uint64, lset labels.Labels) {
|
|||
}
|
||||
|
||||
func (p *MemPostings) addFor(id uint64, l labels.Label) {
|
||||
list := append(p.m[l], id)
|
||||
p.m[l] = list
|
||||
nm, ok := p.m[l.Name]
|
||||
if !ok {
|
||||
nm = map[string][]uint64{}
|
||||
p.m[l.Name] = nm
|
||||
}
|
||||
list := append(nm[l.Value], id)
|
||||
nm[l.Value] = list
|
||||
|
||||
if !p.ordered {
|
||||
return
|
||||
|
|
|
@ -20,21 +20,22 @@ import (
|
|||
"sort"
|
||||
"testing"
|
||||
|
||||
"github.com/prometheus/tsdb/labels"
|
||||
"github.com/prometheus/tsdb/testutil"
|
||||
)
|
||||
|
||||
func TestMemPostings_addFor(t *testing.T) {
|
||||
p := NewMemPostings()
|
||||
p.m[allPostingsKey] = []uint64{1, 2, 3, 4, 6, 7, 8}
|
||||
p.m[allPostingsKey.Name] = map[string][]uint64{}
|
||||
p.m[allPostingsKey.Name][allPostingsKey.Value] = []uint64{1, 2, 3, 4, 6, 7, 8}
|
||||
|
||||
p.addFor(5, allPostingsKey)
|
||||
|
||||
testutil.Equals(t, []uint64{1, 2, 3, 4, 5, 6, 7, 8}, p.m[allPostingsKey])
|
||||
testutil.Equals(t, []uint64{1, 2, 3, 4, 5, 6, 7, 8}, p.m[allPostingsKey.Name][allPostingsKey.Value])
|
||||
}
|
||||
|
||||
func TestMemPostings_ensureOrder(t *testing.T) {
|
||||
p := NewUnorderedMemPostings()
|
||||
p.m["a"] = map[string][]uint64{}
|
||||
|
||||
for i := 0; i < 100; i++ {
|
||||
l := make([]uint64, 100)
|
||||
|
@ -43,12 +44,13 @@ func TestMemPostings_ensureOrder(t *testing.T) {
|
|||
}
|
||||
v := fmt.Sprintf("%d", i)
|
||||
|
||||
p.m[labels.Label{"a", v}] = l
|
||||
p.m["a"][v] = l
|
||||
}
|
||||
|
||||
p.EnsureOrder()
|
||||
|
||||
for _, l := range p.m {
|
||||
for _, e := range p.m {
|
||||
for _, l := range e {
|
||||
ok := sort.SliceIsSorted(l, func(i, j int) bool {
|
||||
return l[i] < l[j]
|
||||
})
|
||||
|
@ -56,6 +58,7 @@ func TestMemPostings_ensureOrder(t *testing.T) {
|
|||
t.Fatalf("postings list %v is not sorted", l)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
type mockPostings struct {
|
||||
|
|
Loading…
Reference in a new issue