Skip to content

Commit

Permalink
Rule Concurrency: Fix debug log
Browse files Browse the repository at this point in the history
The log added here: https://github.com/grafana/mimir/pull/10400/files#diff-3e8efecf11fa9185935eea3d3c9c7993f99422353a391190657877e902875a37R254 doesn't work for all loggers

When I tested it in dev, I got some of these errors:
```
ts=2025-01-10T17:26:37.207249729Z caller=rule_concurrency.go:254 level=debug component=concurrency-controller tenant=17065 group=mimir_querier_api msg="Batched rules into concurrent blocks" batches="unsupported value type"
```

I added a new `fmt.Stringer` type for the slice because I don't want the debug log to negatively affect performance (the added benchmark showed that stringing had a 25% performance hit)
  • Loading branch information
julienduchesne committed Jan 10, 2025
1 parent 91e5fb1 commit 6476312
Show file tree
Hide file tree
Showing 2 changed files with 80 additions and 36 deletions.
13 changes: 12 additions & 1 deletion pkg/ruler/rule_concurrency.go
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ package ruler

import (
"context"
"fmt"
"sync"

"github.com/go-kit/log"
Expand Down Expand Up @@ -178,6 +179,16 @@ func (c *TenantConcurrencyController) Allow(_ context.Context, _ *rules.Group, _
return false
}

// stringableRules is a type that allows us to print a slice of rules.ConcurrentRules.
// This prevents premature evaluation, it will only be evaluated when the logger needs to print it.
type stringableRules []rules.ConcurrentRules

func (p stringableRules) String() string {
return fmt.Sprintf("%v", []rules.ConcurrentRules(p))
}

var _ fmt.Stringer = stringableRules{}

// SplitGroupIntoBatches splits the group into batches of rules that can be evaluated concurrently.
// It tries to batch rules that have no dependencies together and rules that have dependencies in separate batches.
// Returning no batches or nil means that the group should be evaluated sequentially.
Expand Down Expand Up @@ -251,7 +262,7 @@ func (c *TenantConcurrencyController) SplitGroupIntoBatches(_ context.Context, g
}

level.Info(logger).Log("msg", "Batched rules into concurrent blocks", "rules", len(g.Rules()), "batches", len(result))
level.Debug(logger).Log("msg", "Batched rules into concurrent blocks", "batches", result)
level.Debug(logger).Log("msg", "Batched rules into concurrent blocks", "batches", stringableRules(result))

return result
}
Expand Down
103 changes: 68 additions & 35 deletions pkg/ruler/rule_concurrency_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -183,6 +183,40 @@ cortex_ruler_independent_rule_evaluation_concurrency_attempts_completed_total{us
require.True(t, user3ControllerTwo.Allow(ctx, rg, rule1))
}

var splitToBatchesTestCases = map[string]struct {
inputFile string
expectedGroups []rules.ConcurrentRules
}{
"chained": {
inputFile: "fixtures/rules_chain.yaml",
expectedGroups: []rules.ConcurrentRules{
{0, 1},
{2},
{3, 4},
{5, 6},
},
},
"indeterminates": {
inputFile: "fixtures/rules_indeterminates.yaml",
expectedGroups: nil,
},
"all independent": {
inputFile: "fixtures/rules_multiple_independent.yaml",
expectedGroups: []rules.ConcurrentRules{
{0, 1, 2, 3, 4, 5},
},
},
"topological sort": {
inputFile: "fixtures/rules_topological_sort_needed.json",
expectedGroups: []rules.ConcurrentRules{
{0, 4, 8, 12, 16, 20, 24, 28, 32, 36, 37, 38, 58},
{1, 2, 5, 6, 9, 10, 13, 14, 17, 18, 21, 22, 25, 26, 29, 30, 33, 34, 39, 40, 41, 42, 45, 46, 51, 52, 55, 56},
{3, 7, 11, 15, 19, 23, 27, 31, 35},
{43, 44, 47, 48, 49, 50, 53, 54, 57},
},
},
}

func TestSplitGroupIntoBatches(t *testing.T) {
limits := validation.MockOverrides(func(_ *validation.Limits, tenantLimits map[string]*validation.Limits) {
tenantLimits["user1"] = validation.MockDefaultLimits()
Expand All @@ -196,41 +230,7 @@ func TestSplitGroupIntoBatches(t *testing.T) {
RuleConcurrencyController: controller,
})

tests := map[string]struct {
inputFile string
expectedGroups []rules.ConcurrentRules
}{
"chained": {
inputFile: "fixtures/rules_chain.yaml",
expectedGroups: []rules.ConcurrentRules{
{0, 1},
{2},
{3, 4},
{5, 6},
},
},
"indeterminates": {
inputFile: "fixtures/rules_indeterminates.yaml",
expectedGroups: nil,
},
"all independent": {
inputFile: "fixtures/rules_multiple_independent.yaml",
expectedGroups: []rules.ConcurrentRules{
{0, 1, 2, 3, 4, 5},
},
},
"topological sort": {
inputFile: "fixtures/rules_topological_sort_needed.json",
expectedGroups: []rules.ConcurrentRules{
{0, 4, 8, 12, 16, 20, 24, 28, 32, 36, 37, 38, 58},
{1, 2, 5, 6, 9, 10, 13, 14, 17, 18, 21, 22, 25, 26, 29, 30, 33, 34, 39, 40, 41, 42, 45, 46, 51, 52, 55, 56},
{3, 7, 11, 15, 19, 23, 27, 31, 35},
{43, 44, 47, 48, 49, 50, 53, 54, 57},
},
},
}

for name, tc := range tests {
for name, tc := range splitToBatchesTestCases {
t.Run(name, func(t *testing.T) {
// Load group with a -1 interval so it's always at risk.
groups, errs := ruleManager.LoadGroups(-1*time.Second, labels.EmptyLabels(), "", nil, []string{tc.inputFile}...)
Expand All @@ -252,6 +252,39 @@ func TestSplitGroupIntoBatches(t *testing.T) {
}
}

func BenchmarkSplitGroupIntoBatches(b *testing.B) {
limits := validation.MockOverrides(func(_ *validation.Limits, tenantLimits map[string]*validation.Limits) {
tenantLimits["user1"] = validation.MockDefaultLimits()
tenantLimits["user1"].RulerMaxIndependentRuleEvaluationConcurrencyPerTenant = 2
})

mtController := NewMultiTenantConcurrencyController(log.NewNopLogger(), 3, 50.0, prometheus.NewPedanticRegistry(), limits)
controller := mtController.NewTenantConcurrencyControllerFor("user1")

ruleManager := rules.NewManager(&rules.ManagerOptions{
RuleConcurrencyController: controller,
})

for name, tc := range splitToBatchesTestCases {
b.Run(name, func(b *testing.B) {
// Load group with a -1 interval so it's always at risk.
groups, errs := ruleManager.LoadGroups(-1*time.Second, labels.EmptyLabels(), "", nil, []string{tc.inputFile}...)
require.Empty(b, errs)
require.Len(b, groups, 1)

var group *rules.Group
for _, g := range groups {
group = g
}

b.ResetTimer()
for i := 0; i < b.N; i++ {
_ = controller.SplitGroupIntoBatches(context.Background(), group)
}
})
}
}

func requireConcurrentRulesEqual(t *testing.T, expected, actual []rules.ConcurrentRules) {
t.Helper()

Expand Down

0 comments on commit 6476312

Please sign in to comment.