Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Workaround for __name__ grouping in avg() #280

Merged
merged 2 commits into from
Mar 6, 2020
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
47 changes: 47 additions & 0 deletions pkg/proxystorage/proxy.go
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,8 @@ import (
"github.com/jacksontj/promxy/pkg/servergroup"
)

const MetricNameWorkaroundLabel = "__name"

type proxyStorageState struct {
sgs []*servergroup.ServerGroup
client promclient.API
Expand Down Expand Up @@ -296,6 +298,51 @@ func (p *ProxyStorage) NodeReplacer(ctx context.Context, s *promql.EvalStmt, nod

// Convert avg into sum() / count()
case promql.ItemAvg:

nameIncluded := false
for _, g := range n.Grouping {
if g == model.MetricNameLabel {
nameIncluded = true
}
}

if nameIncluded {
replacedGrouping := make([]string, len(n.Grouping))
for i, g := range n.Grouping {
if g == model.MetricNameLabel {
replacedGrouping[i] = MetricNameWorkaroundLabel
} else {
replacedGrouping[i] = g
}
}

return &promql.AggregateExpr{
Op: promql.ItemMax,
Expr: PreserveLabel(&promql.BinaryExpr{
Op: promql.ItemDIV,
LHS: &promql.AggregateExpr{
Op: promql.ItemSum,
Expr: PreserveLabel(CloneExpr(n.Expr), model.MetricNameLabel, MetricNameWorkaroundLabel),
Param: n.Param,
Grouping: replacedGrouping,
Without: n.Without,
},

RHS: &promql.AggregateExpr{
Op: promql.ItemCount,
Expr: PreserveLabel(CloneExpr(n.Expr), model.MetricNameLabel, MetricNameWorkaroundLabel),
Param: n.Param,
Grouping: replacedGrouping,
Without: n.Without,
},
VectorMatching: &promql.VectorMatching{Card: promql.CardOneToOne},
}, MetricNameWorkaroundLabel, model.MetricNameLabel),
Grouping: n.Grouping,
Without: n.Without,
}, nil

}

// Replace with sum() / count()
return &promql.BinaryExpr{
Op: promql.ItemDIV,
Expand Down
6 changes: 6 additions & 0 deletions pkg/proxystorage/util.go
Original file line number Diff line number Diff line change
Expand Up @@ -115,3 +115,9 @@ func CloneExpr(expr promql.Expr) (newExpr promql.Expr) {
newExpr, _ = promql.ParseExpr(expr.String())
return
}

// PreserveLabel wraps the input expression with a label replace in order to preserve the metadata through binary expressions
func PreserveLabel(expr promql.Expr, srcLabel string, dstLabel string) (relabelExpress promql.Expr) {
Copy link
Owner Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'd like to find a way to create this object without having to string and unstring everything if possible (this works, just feels nasty)

Copy link
Owner Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've looked a bit into this and the promql package doesn't make this easy. I also ran a benchmark and it seems that the perf hit is non-existant:

benchmark                                                               old ns/op     new ns/op     delta
BenchmarkEvaluations/benchdata/aggregators.test/direct-4                20803         20102         -3.37%
BenchmarkEvaluations/benchdata/aggregators.test/promxy-4                232304        226409        -2.54%
BenchmarkEvaluations/benchdata/aggregators.test/promxy_remoteread-4     233642        231085        -1.09%

benchmark                                                               old allocs     new allocs     delta
BenchmarkEvaluations/benchdata/aggregators.test/direct-4                143            143            +0.00%
BenchmarkEvaluations/benchdata/aggregators.test/promxy-4                572            571            -0.17%
BenchmarkEvaluations/benchdata/aggregators.test/promxy_remoteread-4     569            569            +0.00%

benchmark                                                               old bytes     new bytes     delta
BenchmarkEvaluations/benchdata/aggregators.test/direct-4                7517          7517          +0.00%
BenchmarkEvaluations/benchdata/aggregators.test/promxy-4                37130         36696         -1.17%
BenchmarkEvaluations/benchdata/aggregators.test/promxy_remoteread-4     36737         36776         +0.11%

So assuming tests pass, I'll rebase/squash and we should be good to go.

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm new to these libraries so I'll continue to dig for a better option. I'm not sure if there is a better way to run the label_replace function as it has to be outside of the aggregate and not inside.

Copy link
Owner Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Besides the fact that this is ugly (String-ing and re-parsing) this seems fine; I was worried that the re-parse would be dramatically slower, but it seems its not noticeable in the scheme of things :)

relabelExpress, _ = promql.ParseExpr(fmt.Sprintf("label_replace(%s,`%s`,`$1`,`%s`,`(.*)`)", expr.String(), dstLabel, srcLabel))
return relabelExpress
}
18 changes: 18 additions & 0 deletions test/testdata/issue_274.test
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
# Test for https://github.com/jacksontj/promxy/issues/274
load 5m
metric_1{a="1"} 1
metric_10{a="10"} 10
metric_1000{a="1000"} 1000
metric_10000{a="10000"} 10000

eval instant at 5m avg({__name__=~"metric_.*"}) by (a)
{a="1"} 1
{a="10"} 10
{a="1000"} 1000
{a="10000"} 10000

eval instant at 5m avg({__name__=~"metric_.*"}) by (__name__)
metric_1{} 1
metric_10{} 10
metric_1000{} 1000
metric_10000{} 10000