mirror of https://github.com/prometheus/prometheus
PromQL: Reduce Code duplication in AST traversion (#6362)
* promql: Add Children function for easier AST traversion Signed-off-by: Tobias Guggenmos <tguggenm@redhat.com>pull/6369/head
parent
66dd5df3d8
commit
d0cff29749
103
promql/ast.go
103
promql/ast.go
|
@ -248,61 +248,10 @@ func Walk(v Visitor, node Node, path []Node) error {
|
||||||
}
|
}
|
||||||
path = append(path, node)
|
path = append(path, node)
|
||||||
|
|
||||||
switch n := node.(type) {
|
for _, e := range Children(node) {
|
||||||
case *EvalStmt:
|
if err := Walk(v, e, path); err != nil {
|
||||||
if err := Walk(v, n.Expr, path); err != nil {
|
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
case Expressions:
|
|
||||||
for _, e := range n {
|
|
||||||
if err := Walk(v, e, path); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
case *AggregateExpr:
|
|
||||||
if n.Param != nil {
|
|
||||||
if err := Walk(v, n.Param, path); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if err := Walk(v, n.Expr, path); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
case *BinaryExpr:
|
|
||||||
if err := Walk(v, n.LHS, path); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := Walk(v, n.RHS, path); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
case *Call:
|
|
||||||
if err := Walk(v, n.Args, path); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
case *SubqueryExpr:
|
|
||||||
if err := Walk(v, n.Expr, path); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
case *ParenExpr:
|
|
||||||
if err := Walk(v, n.Expr, path); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
case *UnaryExpr:
|
|
||||||
if err := Walk(v, n.Expr, path); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
case *MatrixSelector, *NumberLiteral, *StringLiteral, *VectorSelector:
|
|
||||||
// nothing to do
|
|
||||||
|
|
||||||
default:
|
|
||||||
panic(errors.Errorf("promql.Walk: unhandled node type %T", node))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
_, err = v.Visit(nil, nil)
|
_, err = v.Visit(nil, nil)
|
||||||
|
@ -326,3 +275,51 @@ func Inspect(node Node, f inspector) {
|
||||||
//nolint: errcheck
|
//nolint: errcheck
|
||||||
Walk(inspector(f), node, nil)
|
Walk(inspector(f), node, nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Children returns a list of all child nodes of a syntax tree node.
|
||||||
|
func Children(node Node) []Node {
|
||||||
|
// For some reasons these switches have significantly better performance than interfaces
|
||||||
|
switch n := node.(type) {
|
||||||
|
case *EvalStmt:
|
||||||
|
return []Node{n.Expr}
|
||||||
|
case Expressions:
|
||||||
|
// golang cannot convert slices of interfaces
|
||||||
|
ret := make([]Node, len(n))
|
||||||
|
for i, e := range n {
|
||||||
|
ret[i] = e
|
||||||
|
}
|
||||||
|
return ret
|
||||||
|
case *AggregateExpr:
|
||||||
|
// While this does not look nice, it should avoid unnecessary allocations
|
||||||
|
// caused by slice resizing
|
||||||
|
if n.Expr == nil && n.Param == nil {
|
||||||
|
return nil
|
||||||
|
} else if n.Expr == nil {
|
||||||
|
return []Node{n.Param}
|
||||||
|
} else if n.Param == nil {
|
||||||
|
return []Node{n.Expr}
|
||||||
|
} else {
|
||||||
|
return []Node{n.Expr, n.Param}
|
||||||
|
}
|
||||||
|
case *BinaryExpr:
|
||||||
|
return []Node{n.LHS, n.RHS}
|
||||||
|
case *Call:
|
||||||
|
// golang cannot convert slices of interfaces
|
||||||
|
ret := make([]Node, len(n.Args))
|
||||||
|
for i, e := range n.Args {
|
||||||
|
ret[i] = e
|
||||||
|
}
|
||||||
|
return ret
|
||||||
|
case *SubqueryExpr:
|
||||||
|
return []Node{n.Expr}
|
||||||
|
case *ParenExpr:
|
||||||
|
return []Node{n.Expr}
|
||||||
|
case *UnaryExpr:
|
||||||
|
return []Node{n.Expr}
|
||||||
|
case *MatrixSelector, *NumberLiteral, *StringLiteral, *VectorSelector:
|
||||||
|
// nothing to do
|
||||||
|
return []Node{}
|
||||||
|
default:
|
||||||
|
panic(errors.Errorf("promql.Children: unhandled node type %T", node))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -38,39 +38,10 @@ func tree(node Node, level string) string {
|
||||||
|
|
||||||
level += " · · ·"
|
level += " · · ·"
|
||||||
|
|
||||||
switch n := node.(type) {
|
for _, e := range Children(node) {
|
||||||
case *EvalStmt:
|
t += tree(e, level)
|
||||||
t += tree(n.Expr, level)
|
|
||||||
|
|
||||||
case Expressions:
|
|
||||||
for _, e := range n {
|
|
||||||
t += tree(e, level)
|
|
||||||
}
|
|
||||||
case *AggregateExpr:
|
|
||||||
t += tree(n.Expr, level)
|
|
||||||
|
|
||||||
case *BinaryExpr:
|
|
||||||
t += tree(n.LHS, level)
|
|
||||||
t += tree(n.RHS, level)
|
|
||||||
|
|
||||||
case *Call:
|
|
||||||
t += tree(n.Args, level)
|
|
||||||
|
|
||||||
case *ParenExpr:
|
|
||||||
t += tree(n.Expr, level)
|
|
||||||
|
|
||||||
case *UnaryExpr:
|
|
||||||
t += tree(n.Expr, level)
|
|
||||||
|
|
||||||
case *SubqueryExpr:
|
|
||||||
t += tree(n.Expr, level)
|
|
||||||
|
|
||||||
case *MatrixSelector, *NumberLiteral, *StringLiteral, *VectorSelector:
|
|
||||||
// nothing to do
|
|
||||||
|
|
||||||
default:
|
|
||||||
panic("promql.Tree: not all node types covered")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return t
|
return t
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue