2017-07-26 01:34:56 +00:00
|
|
|
package userfunc
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"testing"
|
|
|
|
|
2017-09-11 23:40:37 +00:00
|
|
|
"github.com/hashicorp/hcl2/hcl/hclsyntax"
|
|
|
|
"github.com/hashicorp/hcl2/hcl"
|
2017-07-26 01:34:56 +00:00
|
|
|
"github.com/zclconf/go-cty/cty"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestDecodeUserFunctions(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
src string
|
|
|
|
testExpr string
|
2017-09-11 23:40:37 +00:00
|
|
|
baseCtx *hcl.EvalContext
|
2017-07-26 01:34:56 +00:00
|
|
|
want cty.Value
|
|
|
|
diagCount int
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
`
|
|
|
|
function "greet" {
|
|
|
|
params = ["name"]
|
|
|
|
result = "Hello, ${name}."
|
|
|
|
}
|
|
|
|
`,
|
|
|
|
`greet("Ermintrude")`,
|
|
|
|
nil,
|
|
|
|
cty.StringVal("Hello, Ermintrude."),
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`
|
|
|
|
function "greet" {
|
|
|
|
params = ["name"]
|
|
|
|
result = "Hello, ${name}."
|
|
|
|
}
|
|
|
|
`,
|
|
|
|
`greet()`,
|
|
|
|
nil,
|
|
|
|
cty.DynamicVal,
|
|
|
|
1, // missing value for "name"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`
|
|
|
|
function "greet" {
|
|
|
|
params = ["name"]
|
|
|
|
result = "Hello, ${name}."
|
|
|
|
}
|
|
|
|
`,
|
|
|
|
`greet("Ermintrude", "extra")`,
|
|
|
|
nil,
|
|
|
|
cty.DynamicVal,
|
|
|
|
1, // too many arguments
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`
|
|
|
|
function "add" {
|
|
|
|
params = ["a", "b"]
|
|
|
|
result = a + b
|
|
|
|
}
|
|
|
|
`,
|
|
|
|
`add(1, 5)`,
|
|
|
|
nil,
|
|
|
|
cty.NumberIntVal(6),
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`
|
|
|
|
function "argstuple" {
|
|
|
|
params = []
|
|
|
|
variadic_param = "args"
|
|
|
|
result = args
|
|
|
|
}
|
|
|
|
`,
|
|
|
|
`argstuple("a", true, 1)`,
|
|
|
|
nil,
|
|
|
|
cty.TupleVal([]cty.Value{cty.StringVal("a"), cty.True, cty.NumberIntVal(1)}),
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`
|
|
|
|
function "missing_var" {
|
|
|
|
params = []
|
|
|
|
result = nonexist
|
|
|
|
}
|
|
|
|
`,
|
|
|
|
`missing_var()`,
|
|
|
|
nil,
|
|
|
|
cty.DynamicVal,
|
|
|
|
1, // no variable named "nonexist"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`
|
|
|
|
function "closure" {
|
|
|
|
params = []
|
|
|
|
result = upvalue
|
|
|
|
}
|
|
|
|
`,
|
|
|
|
`closure()`,
|
2017-09-11 23:40:37 +00:00
|
|
|
&hcl.EvalContext{
|
2017-07-26 01:34:56 +00:00
|
|
|
Variables: map[string]cty.Value{
|
|
|
|
"upvalue": cty.True,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
cty.True,
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`
|
|
|
|
function "neg" {
|
|
|
|
params = ["val"]
|
|
|
|
result = -val
|
|
|
|
}
|
|
|
|
function "add" {
|
|
|
|
params = ["a", "b"]
|
|
|
|
result = a + b
|
|
|
|
}
|
|
|
|
`,
|
|
|
|
`neg(add(1, 3))`,
|
|
|
|
nil,
|
|
|
|
cty.NumberIntVal(-4),
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`
|
|
|
|
function "neg" {
|
|
|
|
parrams = ["val"]
|
|
|
|
result = -val
|
|
|
|
}
|
|
|
|
`,
|
|
|
|
`null`,
|
|
|
|
nil,
|
|
|
|
cty.NullVal(cty.DynamicPseudoType),
|
|
|
|
2, // missing attribute "params", and unknown attribute "parrams"
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, test := range tests {
|
|
|
|
t.Run(fmt.Sprintf("%02d", i), func(t *testing.T) {
|
2017-09-11 23:40:37 +00:00
|
|
|
f, diags := hclsyntax.ParseConfig([]byte(test.src), "config", hcl.Pos{Line: 1, Column: 1})
|
2017-07-26 01:34:56 +00:00
|
|
|
if f == nil || f.Body == nil {
|
|
|
|
t.Fatalf("got nil file or body")
|
|
|
|
}
|
|
|
|
|
2017-09-11 23:40:37 +00:00
|
|
|
funcs, _, funcsDiags := decodeUserFunctions(f.Body, "function", func() *hcl.EvalContext {
|
2017-07-26 01:34:56 +00:00
|
|
|
return test.baseCtx
|
|
|
|
})
|
|
|
|
diags = append(diags, funcsDiags...)
|
|
|
|
|
2017-09-11 23:40:37 +00:00
|
|
|
expr, exprParseDiags := hclsyntax.ParseExpression([]byte(test.testExpr), "testexpr", hcl.Pos{Line: 1, Column: 1})
|
2017-07-26 01:34:56 +00:00
|
|
|
diags = append(diags, exprParseDiags...)
|
|
|
|
if expr == nil {
|
|
|
|
t.Fatalf("parsing test expr returned nil")
|
|
|
|
}
|
|
|
|
|
2017-09-11 23:40:37 +00:00
|
|
|
got, exprDiags := expr.Value(&hcl.EvalContext{
|
2017-07-26 01:34:56 +00:00
|
|
|
Functions: funcs,
|
|
|
|
})
|
|
|
|
diags = append(diags, exprDiags...)
|
|
|
|
|
|
|
|
if len(diags) != test.diagCount {
|
|
|
|
t.Errorf("wrong number of diagnostics %d; want %d", len(diags), test.diagCount)
|
|
|
|
for _, diag := range diags {
|
|
|
|
t.Logf("- %s", diag)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !got.RawEquals(test.want) {
|
|
|
|
t.Errorf("wrong result\ngot: %#v\nwant: %#v", got, test.want)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|