mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2025-01-20 15:16:42 +00:00
209a5024ce
Previously, map for storing tenant metrics was re-created to each newly ingested tenant. It has significant performance impact for systems with large number of tenants. This commit addresses this issue by changing algorithm of creating tenant metric records at map. Instead of map re-creation, it uses `sync.Map` primitive. Benchmark results: ``` goos: linux goarch: amd64 pkg: github.com/VictoriaMetrics/VictoriaMetrics/lib/tenantmetrics cpu: AMD Ryzen 9 5900X 12-Core Processor │ lib/tenantmetrics/orig.bench │ lib/tenantmetrics/new.bench │ │ sec/op │ sec/op vs base │ CounterMapGrowth/n=100,nProcs=GOMAXPROCS-24 1943.2µ ± 5% 248.0µ ± 11% -87.24% (p=0.001 n=7) CounterMapGrowth/n=100-24 434.63µ ± 5% 98.82µ ± 16% -77.26% (p=0.001 n=7) CounterMapGrowth/n=1000-24 32.719m ± 20% 1.425m ± 5% -95.65% (p=0.001 n=7) CounterMapGrowth/n=10000-24 3653.60m ± 5% 18.00m ± 2% -99.51% (p=0.001 n=7) geomean 17.83m 890.4µ -95.00% ``` Related issue: https://github.com/VictoriaMetrics/VictoriaMetrics/issues/7482 --- Co-authored-by: Artem Fetishev <rtm@victoriametrics.com>
46 lines
828 B
Go
46 lines
828 B
Go
package tenantmetrics
|
|
|
|
import (
|
|
"runtime"
|
|
"sync"
|
|
"testing"
|
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/auth"
|
|
)
|
|
|
|
func BenchmarkCounterMapGrowth(b *testing.B) {
|
|
f := func(b *testing.B, numTenants uint32, nProcs int) {
|
|
b.Helper()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
cm := NewCounterMap("foobar")
|
|
var wg sync.WaitGroup
|
|
for range nProcs {
|
|
wg.Add(1)
|
|
go func() {
|
|
for i := range numTenants {
|
|
cm.Get(&auth.Token{AccountID: i, ProjectID: i}).Inc()
|
|
}
|
|
wg.Done()
|
|
}()
|
|
}
|
|
wg.Wait()
|
|
}
|
|
}
|
|
|
|
b.Run("n=100,nProcs=GOMAXPROCS", func(b *testing.B) {
|
|
f(b, 100, runtime.GOMAXPROCS(0))
|
|
})
|
|
|
|
b.Run("n=100,nProcs=2", func(b *testing.B) {
|
|
f(b, 100, 2)
|
|
})
|
|
|
|
b.Run("n=1000,nProcs=2", func(b *testing.B) {
|
|
f(b, 1000, 2)
|
|
})
|
|
|
|
b.Run("n=10000,nProcs=2", func(b *testing.B) {
|
|
f(b, 10000, 2)
|
|
})
|
|
}
|