2017-05-25 15:14:43 +00:00
|
|
|
package zclsyntax
|
|
|
|
|
|
|
|
import (
|
|
|
|
"testing"
|
|
|
|
|
2017-05-28 00:35:44 +00:00
|
|
|
"github.com/zclconf/go-cty/cty"
|
|
|
|
"github.com/zclconf/go-cty/cty/function"
|
|
|
|
"github.com/zclconf/go-cty/cty/function/stdlib"
|
2017-05-28 00:33:09 +00:00
|
|
|
"github.com/zclconf/go-zcl/zcl"
|
2017-05-25 15:14:43 +00:00
|
|
|
)
|
|
|
|
|
2017-06-01 02:27:16 +00:00
|
|
|
func TestExpressionParseAndValue(t *testing.T) {
|
|
|
|
// This is a combo test that exercises both the parser and the Value
|
|
|
|
// method, with the focus on the latter but indirectly testing the former.
|
|
|
|
tests := []struct {
|
|
|
|
input string
|
|
|
|
ctx *zcl.EvalContext
|
|
|
|
want cty.Value
|
|
|
|
diagCount int
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
`1`,
|
|
|
|
nil,
|
|
|
|
cty.NumberIntVal(1),
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`(1)`,
|
|
|
|
nil,
|
|
|
|
cty.NumberIntVal(1),
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`(1`,
|
|
|
|
nil,
|
|
|
|
cty.NumberIntVal(1),
|
|
|
|
1, // Unbalanced parentheses
|
|
|
|
},
|
2017-06-01 13:56:45 +00:00
|
|
|
{
|
|
|
|
`true`,
|
|
|
|
nil,
|
|
|
|
cty.True,
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`false`,
|
|
|
|
nil,
|
|
|
|
cty.False,
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`null`,
|
|
|
|
nil,
|
|
|
|
cty.NullVal(cty.DynamicPseudoType),
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`true true`,
|
|
|
|
nil,
|
|
|
|
cty.True,
|
|
|
|
1, // extra characters after expression
|
|
|
|
},
|
2017-06-01 15:01:12 +00:00
|
|
|
{
|
|
|
|
`"hello"`,
|
|
|
|
nil,
|
|
|
|
cty.StringVal("hello"),
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`"hello\nworld"`,
|
|
|
|
nil,
|
|
|
|
cty.StringVal("hello\nworld"),
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`"unclosed`,
|
|
|
|
nil,
|
|
|
|
cty.StringVal("unclosed"),
|
|
|
|
1, // Unterminated template string
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`"hello ${"world"}"`,
|
|
|
|
nil,
|
|
|
|
cty.StringVal("hello world"),
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`"hello ${12.5}"`,
|
|
|
|
nil,
|
|
|
|
cty.StringVal("hello 12.5"),
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`"silly ${"${"nesting"}"}"`,
|
|
|
|
nil,
|
|
|
|
cty.StringVal("silly nesting"),
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`"silly ${"${true}"}"`,
|
|
|
|
nil,
|
|
|
|
cty.StringVal("silly true"),
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`"hello $${escaped}"`,
|
|
|
|
nil,
|
|
|
|
cty.StringVal("hello ${escaped}"),
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
`"hello $$nonescape"`,
|
|
|
|
nil,
|
|
|
|
cty.StringVal("hello $$nonescape"),
|
|
|
|
0,
|
|
|
|
},
|
2017-06-01 02:27:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range tests {
|
|
|
|
t.Run(test.input, func(t *testing.T) {
|
|
|
|
expr, parseDiags := ParseExpression([]byte(test.input), "", zcl.Pos{Line: 1, Column: 1, Byte: 0})
|
|
|
|
|
|
|
|
got, valDiags := expr.Value(test.ctx)
|
|
|
|
|
|
|
|
diagCount := len(parseDiags) + len(valDiags)
|
|
|
|
|
|
|
|
if diagCount != test.diagCount {
|
|
|
|
t.Errorf("wrong number of diagnostics %d; want %d", diagCount, test.diagCount)
|
|
|
|
for _, diag := range parseDiags {
|
|
|
|
t.Logf(" - %s", diag.Error())
|
|
|
|
}
|
|
|
|
for _, diag := range valDiags {
|
|
|
|
t.Logf(" - %s", diag.Error())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !got.RawEquals(test.want) {
|
|
|
|
t.Errorf("wrong result\ngot: %#v\nwant: %#v", got, test.want)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2017-05-25 15:14:43 +00:00
|
|
|
func TestFunctionCallExprValue(t *testing.T) {
|
|
|
|
funcs := map[string]function.Function{
|
|
|
|
"length": stdlib.StrlenFunc,
|
|
|
|
"jsondecode": stdlib.JSONDecodeFunc,
|
|
|
|
}
|
|
|
|
|
|
|
|
tests := map[string]struct {
|
|
|
|
expr *FunctionCallExpr
|
|
|
|
ctx *zcl.EvalContext
|
|
|
|
want cty.Value
|
|
|
|
diagCount int
|
|
|
|
}{
|
|
|
|
"valid call with no conversions": {
|
|
|
|
&FunctionCallExpr{
|
|
|
|
Name: "length",
|
|
|
|
Args: []Expression{
|
|
|
|
&LiteralValueExpr{
|
|
|
|
Val: cty.StringVal("hello"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&zcl.EvalContext{
|
|
|
|
Functions: funcs,
|
|
|
|
},
|
|
|
|
cty.NumberIntVal(5),
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
"valid call with arg conversion": {
|
|
|
|
&FunctionCallExpr{
|
|
|
|
Name: "length",
|
|
|
|
Args: []Expression{
|
|
|
|
&LiteralValueExpr{
|
|
|
|
Val: cty.BoolVal(true),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&zcl.EvalContext{
|
|
|
|
Functions: funcs,
|
|
|
|
},
|
|
|
|
cty.NumberIntVal(4), // length of string "true"
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
"valid call with unknown arg": {
|
|
|
|
&FunctionCallExpr{
|
|
|
|
Name: "length",
|
|
|
|
Args: []Expression{
|
|
|
|
&LiteralValueExpr{
|
|
|
|
Val: cty.UnknownVal(cty.String),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&zcl.EvalContext{
|
|
|
|
Functions: funcs,
|
|
|
|
},
|
|
|
|
cty.UnknownVal(cty.Number),
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
"valid call with unknown arg needing conversion": {
|
|
|
|
&FunctionCallExpr{
|
|
|
|
Name: "length",
|
|
|
|
Args: []Expression{
|
|
|
|
&LiteralValueExpr{
|
|
|
|
Val: cty.UnknownVal(cty.Bool),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&zcl.EvalContext{
|
|
|
|
Functions: funcs,
|
|
|
|
},
|
|
|
|
cty.UnknownVal(cty.Number),
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
"valid call with dynamic arg": {
|
|
|
|
&FunctionCallExpr{
|
|
|
|
Name: "length",
|
|
|
|
Args: []Expression{
|
|
|
|
&LiteralValueExpr{
|
|
|
|
Val: cty.DynamicVal,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&zcl.EvalContext{
|
|
|
|
Functions: funcs,
|
|
|
|
},
|
|
|
|
cty.UnknownVal(cty.Number),
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
"invalid arg type": {
|
|
|
|
&FunctionCallExpr{
|
|
|
|
Name: "length",
|
|
|
|
Args: []Expression{
|
|
|
|
&LiteralValueExpr{
|
|
|
|
Val: cty.ListVal([]cty.Value{cty.StringVal("hello")}),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&zcl.EvalContext{
|
|
|
|
Functions: funcs,
|
|
|
|
},
|
|
|
|
cty.DynamicVal,
|
|
|
|
1,
|
|
|
|
},
|
|
|
|
"function with dynamic return type": {
|
|
|
|
&FunctionCallExpr{
|
|
|
|
Name: "jsondecode",
|
|
|
|
Args: []Expression{
|
|
|
|
&LiteralValueExpr{
|
|
|
|
Val: cty.StringVal(`"hello"`),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&zcl.EvalContext{
|
|
|
|
Functions: funcs,
|
|
|
|
},
|
|
|
|
cty.StringVal("hello"),
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
"function with dynamic return type unknown arg": {
|
|
|
|
&FunctionCallExpr{
|
|
|
|
Name: "jsondecode",
|
|
|
|
Args: []Expression{
|
|
|
|
&LiteralValueExpr{
|
|
|
|
Val: cty.UnknownVal(cty.String),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&zcl.EvalContext{
|
|
|
|
Functions: funcs,
|
|
|
|
},
|
|
|
|
cty.DynamicVal, // type depends on arg value
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
"error in function": {
|
|
|
|
&FunctionCallExpr{
|
|
|
|
Name: "jsondecode",
|
|
|
|
Args: []Expression{
|
|
|
|
&LiteralValueExpr{
|
|
|
|
Val: cty.StringVal("invalid-json"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&zcl.EvalContext{
|
|
|
|
Functions: funcs,
|
|
|
|
},
|
|
|
|
cty.DynamicVal,
|
|
|
|
1, // JSON parse error
|
|
|
|
},
|
|
|
|
"unknown function": {
|
|
|
|
&FunctionCallExpr{
|
|
|
|
Name: "lenth",
|
|
|
|
Args: []Expression{},
|
|
|
|
},
|
|
|
|
&zcl.EvalContext{
|
|
|
|
Functions: funcs,
|
|
|
|
},
|
|
|
|
cty.DynamicVal,
|
|
|
|
1,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for name, test := range tests {
|
|
|
|
t.Run(name, func(t *testing.T) {
|
|
|
|
got, diags := test.expr.Value(test.ctx)
|
|
|
|
|
|
|
|
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.Error())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !got.RawEquals(test.want) {
|
|
|
|
t.Errorf("wrong result\ngot: %#v\nwant: %#v", got, test.want)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|