hcl/ext/dynblock/expr_wrap.go
Martin Atkins da95646a33 ext/dynblock: dynamic blocks extension
This extension allows an application to support dynamic generation of
child blocks based on expressions in certain contexts. This is done using
a new block type called "dynamic", which contains an iteration value
(which must be a collection) and a specification of how to construct a
child block for each element of that collection.
2018-01-27 09:10:18 -08:00

61 lines
1.4 KiB
Go

package dynblock
import (
"github.com/hashicorp/hcl2/hcl"
"github.com/zclconf/go-cty/cty"
)
type exprWrap struct {
hcl.Expression
i *iteration
}
func (e exprWrap) Variables() []hcl.Traversal {
raw := e.Expression.Variables()
ret := make([]hcl.Traversal, 0, len(raw))
// Filter out traversals that refer to our iterator name or any
// iterator we've inherited; we're going to provide those in
// our Value wrapper, so the caller doesn't need to know about them.
for _, traversal := range raw {
rootName := traversal.RootName()
if rootName == e.i.IteratorName {
continue
}
if _, inherited := e.i.Inherited[rootName]; inherited {
continue
}
ret = append(ret, traversal)
}
return ret
}
func (e exprWrap) Value(ctx *hcl.EvalContext) (cty.Value, hcl.Diagnostics) {
extCtx := e.i.EvalContext(ctx)
return e.Expression.Value(extCtx)
}
// Passthrough implementation for hcl.ExprList
func (e exprWrap) ExprList() []hcl.Expression {
type exprList interface {
ExprList() []hcl.Expression
}
if el, supported := e.Expression.(exprList); supported {
return el.ExprList()
}
return nil
}
// Passthrough implementation for hcl.AbsTraversalForExpr and hcl.RelTraversalForExpr
func (e exprWrap) AsTraversal() hcl.Traversal {
type asTraversal interface {
AsTraversal() hcl.Traversal
}
if at, supported := e.Expression.(asTraversal); supported {
return at.AsTraversal()
}
return nil
}