/* Copyright 2020 Google LLC Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at https://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ package build // StopTraversalError is a special error that tells the walker to not traverse // further and visit child nodes of the current node. type StopTraversalError struct{} func (m *StopTraversalError) Error() string { return "Stop traversal" } // Walk walks the expression tree v, calling f on all subexpressions // in a preorder traversal. // // The stk argument is the stack of expressions in the recursion above x, // from outermost to innermost. // func Walk(v Expr, f func(x Expr, stk []Expr)) { var stack []Expr walk1(&v, &stack, func(x *Expr, stk []Expr) (Expr, error) { f(*x, stk) return nil, nil }) } // WalkPointers is the same as Walk but calls the callback function with pointers to nodes. func WalkPointers(v Expr, f func(x *Expr, stk []Expr)) { var stack []Expr walk1(&v, &stack, func(x *Expr, stk []Expr) (Expr, error) { f(x, stk) return nil, nil }) } // WalkInterruptable is the same as Walk but allows traversal to be interrupted. func WalkInterruptable(v Expr, f func(x Expr, stk []Expr) error) { var stack []Expr walk1(&v, &stack, func(x *Expr, stk []Expr) (Expr, error) { err := f(*x, stk) return nil, err }) } // Edit walks the expression tree v, calling f on all subexpressions // in a preorder traversal. If f returns a non-nil value, the tree is mutated. // The new value replaces the old one. // // The stk argument is the stack of expressions in the recursion above x, // from outermost to innermost. // func Edit(v Expr, f func(x Expr, stk []Expr) Expr) Expr { var stack []Expr return walk1(&v, &stack, func(x *Expr, stk []Expr) (Expr, error) { return f(*x, stk), nil }) } // EditChildren is similar to Edit but doesn't visit the initial node, instead goes // directly to its children. func EditChildren(v Expr, f func(x Expr, stk []Expr) Expr) { stack := []Expr{v} WalkOnce(v, func(x *Expr) { walk1(x, &stack, func(x *Expr, stk []Expr) (Expr, error) { return f(*x, stk), nil }) }) } // walk1 is a helper function for Walk, WalkWithPostfix, and Edit. func walk1(v *Expr, stack *[]Expr, f func(x *Expr, stk []Expr) (Expr, error)) Expr { if v == nil || *v == nil { return nil } res, err := f(v, *stack) if res != nil { *v = res } if err != nil { if _, ok := err.(*StopTraversalError); ok { // Don't traverse inside return nil } } *stack = append(*stack, *v) WalkOnce(*v, func(x *Expr) { walk1(x, stack, f) }) *stack = (*stack)[:len(*stack)-1] return *v } // WalkOnce calls f on every child of v. func WalkOnce(v Expr, f func(x *Expr)) { switch v := v.(type) { case *File: for i := range v.Stmt { f(&v.Stmt[i]) } case *DotExpr: f(&v.X) case *IndexExpr: f(&v.X) f(&v.Y) case *KeyValueExpr: f(&v.Key) f(&v.Value) case *SliceExpr: f(&v.X) if v.From != nil { f(&v.From) } if v.To != nil { f(&v.To) } if v.Step != nil { f(&v.Step) } case *ParenExpr: f(&v.X) case *UnaryExpr: f(&v.X) case *BinaryExpr: f(&v.X) f(&v.Y) case *AssignExpr: f(&v.LHS) f(&v.RHS) case *LambdaExpr: for i := range v.Params { f(&v.Params[i]) } for i := range v.Body { f(&v.Body[i]) } case *CallExpr: f(&v.X) for i := range v.List { f(&v.List[i]) } case *ListExpr: for i := range v.List { f(&v.List[i]) } case *SetExpr: for i := range v.List { f(&v.List[i]) } case *TupleExpr: for i := range v.List { f(&v.List[i]) } case *DictExpr: for i := range v.List { e := Expr(v.List[i]) f(&e) } case *Comprehension: f(&v.Body) for _, c := range v.Clauses { f(&c) } case *IfClause: f(&v.Cond) case *ForClause: f(&v.Vars) f(&v.X) case *ConditionalExpr: f(&v.Then) f(&v.Test) f(&v.Else) case *LoadStmt: module := (Expr)(v.Module) f(&module) v.Module = module.(*StringExpr) for i := range v.From { from := (Expr)(v.From[i]) f(&from) v.From[i] = from.(*Ident) to := (Expr)(v.To[i]) f(&to) v.To[i] = to.(*Ident) } case *DefStmt: for i := range v.Params { f(&v.Params[i]) } for i := range v.Body { f(&v.Body[i]) } case *IfStmt: f(&v.Cond) for i := range v.True { f(&v.True[i]) } for i := range v.False { f(&v.False[i]) } case *ForStmt: f(&v.Vars) f(&v.X) for i := range v.Body { f(&v.Body[i]) } case *ReturnStmt: if v.Result != nil { f(&v.Result) } } } // walkStatements is a helper function for WalkStatements func walkStatements(v Expr, stack *[]Expr, f func(x Expr, stk []Expr) error) { if v == nil { return } err := f(v, *stack) if err != nil { if _, ok := err.(*StopTraversalError); ok { return } } *stack = append(*stack, v) traverse := func(x Expr) { walkStatements(x, stack, f) } switch expr := v.(type) { case *File: for _, s := range expr.Stmt { traverse(s) } case *DefStmt: for _, s := range expr.Body { traverse(s) } case *IfStmt: for _, s := range expr.True { traverse(s) } for _, s := range expr.False { traverse(s) } case *ForStmt: for _, s := range expr.Body { traverse(s) } } *stack = (*stack)[:len(*stack)-1] } // WalkStatements traverses sub statements (not all nodes) func WalkStatements(v Expr, f func(x Expr, stk []Expr) error) { var stack []Expr walkStatements(v, &stack, f) }