2017-05-24 15:05:52 +00:00
|
|
|
package zclsyntax
|
|
|
|
|
|
|
|
import (
|
2017-05-28 00:33:09 +00:00
|
|
|
"github.com/zclconf/go-zcl/zcl"
|
2017-05-24 15:05:52 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// Variables returns all of the variables referenced within a given experssion.
|
|
|
|
//
|
|
|
|
// This is the implementation of the "Variables" method on every native
|
|
|
|
// expression.
|
|
|
|
func Variables(expr Expression) []zcl.Traversal {
|
|
|
|
var vars []zcl.Traversal
|
2017-06-13 15:53:33 +00:00
|
|
|
|
2017-06-14 15:03:32 +00:00
|
|
|
walker := &variablesWalker{
|
|
|
|
Callback: func(t zcl.Traversal) {
|
|
|
|
vars = append(vars, t)
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
Walk(expr, walker)
|
2017-06-13 15:53:33 +00:00
|
|
|
|
2017-05-24 15:05:52 +00:00
|
|
|
return vars
|
|
|
|
}
|
2017-06-14 15:03:32 +00:00
|
|
|
|
|
|
|
// variablesWalker is a Walker implementation that calls its callback for any
|
|
|
|
// root scope traversal found while walking.
|
|
|
|
type variablesWalker struct {
|
|
|
|
Callback func(zcl.Traversal)
|
|
|
|
localScopes []map[string]struct{}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *variablesWalker) Enter(n Node) zcl.Diagnostics {
|
|
|
|
switch tn := n.(type) {
|
|
|
|
case *ScopeTraversalExpr:
|
|
|
|
t := tn.Traversal
|
|
|
|
|
|
|
|
// Check if the given root name appears in any of the active
|
|
|
|
// local scopes. We don't want to return local variables here, since
|
|
|
|
// the goal of walking variables is to tell the calling application
|
|
|
|
// which names it needs to populate in the _root_ scope.
|
|
|
|
name := t.RootName()
|
|
|
|
for _, names := range w.localScopes {
|
|
|
|
if _, localized := names[name]; localized {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
w.Callback(t)
|
|
|
|
case ChildScope:
|
|
|
|
w.localScopes = append(w.localScopes, tn.LocalNames)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *variablesWalker) Exit(n Node) zcl.Diagnostics {
|
|
|
|
switch n.(type) {
|
|
|
|
case ChildScope:
|
|
|
|
// pop the latest local scope, assuming that the walker will
|
|
|
|
// behave symmetrically as promised.
|
|
|
|
w.localScopes = w.localScopes[:len(w.localScopes)-1]
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ChildScope is a synthetic AST node that is visited during a walk to
|
|
|
|
// indicate that its descendent will be evaluated in a child scope, which
|
|
|
|
// may mask certain variables from the parent scope as locals.
|
|
|
|
//
|
|
|
|
// ChildScope nodes don't really exist in the AST, but are rather synthesized
|
|
|
|
// on the fly during walk. Therefore it doesn't do any good to transform them;
|
|
|
|
// instead, transform either parent node that created a scope or the expression
|
|
|
|
// that the child scope struct wraps.
|
|
|
|
type ChildScope struct {
|
|
|
|
LocalNames map[string]struct{}
|
|
|
|
Expr *Expression // pointer because it can be replaced on walk
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e ChildScope) walkChildNodes(w internalWalkFunc) {
|
|
|
|
*(e.Expr) = w(*(e.Expr)).(Expression)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Range returns the range of the expression that the ChildScope is
|
|
|
|
// encapsulating. It isn't really very useful to call Range on a ChildScope.
|
|
|
|
func (e ChildScope) Range() zcl.Range {
|
|
|
|
return (*e.Expr).Range()
|
|
|
|
}
|