2017-06-03 14:29:48 +00:00
|
|
|
package zclsyntax
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"reflect"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/kylelemons/godebug/pretty"
|
|
|
|
"github.com/zclconf/go-cty/cty"
|
|
|
|
"github.com/zclconf/go-zcl/zcl"
|
|
|
|
)
|
|
|
|
|
2017-06-03 16:04:16 +00:00
|
|
|
func TestBodyContent(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
body *Body
|
|
|
|
schema *zcl.BodySchema
|
|
|
|
partial bool
|
|
|
|
want *zcl.BodyContent
|
|
|
|
diagCount int
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
&Body{},
|
|
|
|
&zcl.BodySchema{},
|
|
|
|
false,
|
|
|
|
&zcl.BodyContent{
|
|
|
|
Attributes: zcl.Attributes{},
|
|
|
|
},
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
|
|
|
|
// Attributes
|
|
|
|
{
|
|
|
|
&Body{
|
|
|
|
Attributes: Attributes{
|
|
|
|
"foo": &Attribute{
|
|
|
|
Name: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&zcl.BodySchema{
|
|
|
|
Attributes: []zcl.AttributeSchema{
|
|
|
|
{
|
|
|
|
Name: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
&zcl.BodyContent{
|
|
|
|
Attributes: zcl.Attributes{
|
|
|
|
"foo": &zcl.Attribute{
|
|
|
|
Name: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
&Body{
|
|
|
|
Attributes: Attributes{
|
|
|
|
"foo": &Attribute{
|
|
|
|
Name: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&zcl.BodySchema{},
|
|
|
|
false,
|
|
|
|
&zcl.BodyContent{
|
|
|
|
Attributes: zcl.Attributes{},
|
|
|
|
},
|
|
|
|
1, // attribute "foo" is not expected
|
|
|
|
},
|
|
|
|
{
|
|
|
|
&Body{
|
|
|
|
Attributes: Attributes{
|
|
|
|
"foo": &Attribute{
|
|
|
|
Name: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&zcl.BodySchema{},
|
|
|
|
true,
|
|
|
|
&zcl.BodyContent{
|
|
|
|
Attributes: zcl.Attributes{},
|
|
|
|
},
|
|
|
|
0, // in partial mode, so extra "foo" is acceptable
|
|
|
|
},
|
|
|
|
{
|
|
|
|
&Body{
|
|
|
|
Attributes: Attributes{},
|
|
|
|
},
|
|
|
|
&zcl.BodySchema{
|
|
|
|
Attributes: []zcl.AttributeSchema{
|
|
|
|
{
|
|
|
|
Name: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
&zcl.BodyContent{
|
|
|
|
Attributes: zcl.Attributes{},
|
|
|
|
},
|
|
|
|
0, // "foo" not required, so no error
|
|
|
|
},
|
|
|
|
{
|
|
|
|
&Body{
|
|
|
|
Attributes: Attributes{},
|
|
|
|
},
|
|
|
|
&zcl.BodySchema{
|
|
|
|
Attributes: []zcl.AttributeSchema{
|
|
|
|
{
|
|
|
|
Name: "foo",
|
|
|
|
Required: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
&zcl.BodyContent{
|
|
|
|
Attributes: zcl.Attributes{},
|
|
|
|
},
|
|
|
|
1, // "foo" is required
|
|
|
|
},
|
|
|
|
{
|
|
|
|
&Body{
|
|
|
|
Attributes: Attributes{
|
|
|
|
"foo": &Attribute{
|
|
|
|
Name: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&zcl.BodySchema{
|
|
|
|
Blocks: []zcl.BlockHeaderSchema{
|
|
|
|
{
|
|
|
|
Type: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
&zcl.BodyContent{
|
|
|
|
Attributes: zcl.Attributes{},
|
|
|
|
},
|
|
|
|
1, // attribute "foo" not expected (it's defined as a block)
|
|
|
|
},
|
|
|
|
|
|
|
|
// Blocks
|
|
|
|
{
|
|
|
|
&Body{
|
|
|
|
Blocks: Blocks{
|
|
|
|
&Block{
|
|
|
|
Type: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&zcl.BodySchema{
|
|
|
|
Blocks: []zcl.BlockHeaderSchema{
|
|
|
|
{
|
|
|
|
Type: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
&zcl.BodyContent{
|
|
|
|
Attributes: zcl.Attributes{},
|
|
|
|
Blocks: zcl.Blocks{
|
|
|
|
{
|
|
|
|
Type: "foo",
|
|
|
|
Body: (*Body)(nil),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
&Body{
|
|
|
|
Blocks: Blocks{
|
|
|
|
&Block{
|
|
|
|
Type: "foo",
|
|
|
|
},
|
|
|
|
&Block{
|
|
|
|
Type: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&zcl.BodySchema{
|
|
|
|
Blocks: []zcl.BlockHeaderSchema{
|
|
|
|
{
|
|
|
|
Type: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
&zcl.BodyContent{
|
|
|
|
Attributes: zcl.Attributes{},
|
|
|
|
Blocks: zcl.Blocks{
|
|
|
|
{
|
|
|
|
Type: "foo",
|
|
|
|
Body: (*Body)(nil),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Type: "foo",
|
|
|
|
Body: (*Body)(nil),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
&Body{
|
|
|
|
Blocks: Blocks{
|
|
|
|
&Block{
|
|
|
|
Type: "foo",
|
|
|
|
},
|
|
|
|
&Block{
|
|
|
|
Type: "bar",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&zcl.BodySchema{
|
|
|
|
Blocks: []zcl.BlockHeaderSchema{
|
|
|
|
{
|
|
|
|
Type: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
&zcl.BodyContent{
|
|
|
|
Attributes: zcl.Attributes{},
|
|
|
|
Blocks: zcl.Blocks{
|
|
|
|
{
|
|
|
|
Type: "foo",
|
|
|
|
Body: (*Body)(nil),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
1, // blocks of type "bar" not expected
|
|
|
|
},
|
|
|
|
{
|
|
|
|
&Body{
|
|
|
|
Blocks: Blocks{
|
|
|
|
&Block{
|
|
|
|
Type: "foo",
|
|
|
|
},
|
|
|
|
&Block{
|
|
|
|
Type: "bar",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&zcl.BodySchema{
|
|
|
|
Blocks: []zcl.BlockHeaderSchema{
|
|
|
|
{
|
|
|
|
Type: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
true,
|
|
|
|
&zcl.BodyContent{
|
|
|
|
Attributes: zcl.Attributes{},
|
|
|
|
Blocks: zcl.Blocks{
|
|
|
|
{
|
|
|
|
Type: "foo",
|
|
|
|
Body: (*Body)(nil),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
0, // extra "bar" allowed because we're in partial mode
|
|
|
|
},
|
|
|
|
{
|
|
|
|
&Body{
|
|
|
|
Blocks: Blocks{
|
|
|
|
&Block{
|
|
|
|
Type: "foo",
|
|
|
|
Labels: []string{"bar"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&zcl.BodySchema{
|
|
|
|
Blocks: []zcl.BlockHeaderSchema{
|
|
|
|
{
|
|
|
|
Type: "foo",
|
|
|
|
LabelNames: []string{"name"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
&zcl.BodyContent{
|
|
|
|
Attributes: zcl.Attributes{},
|
|
|
|
Blocks: zcl.Blocks{
|
|
|
|
{
|
|
|
|
Type: "foo",
|
|
|
|
Labels: []string{"bar"},
|
|
|
|
Body: (*Body)(nil),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
&Body{
|
|
|
|
Blocks: Blocks{
|
|
|
|
&Block{
|
|
|
|
Type: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&zcl.BodySchema{
|
|
|
|
Blocks: []zcl.BlockHeaderSchema{
|
|
|
|
{
|
|
|
|
Type: "foo",
|
|
|
|
LabelNames: []string{"name"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
&zcl.BodyContent{
|
|
|
|
Attributes: zcl.Attributes{},
|
|
|
|
},
|
|
|
|
1, // missing label "name"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
&Body{
|
|
|
|
Blocks: Blocks{
|
|
|
|
&Block{
|
|
|
|
Type: "foo",
|
|
|
|
Labels: []string{"bar"},
|
|
|
|
|
|
|
|
LabelRanges: []zcl.Range{{}},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&zcl.BodySchema{
|
|
|
|
Blocks: []zcl.BlockHeaderSchema{
|
|
|
|
{
|
|
|
|
Type: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
&zcl.BodyContent{
|
|
|
|
Attributes: zcl.Attributes{},
|
|
|
|
},
|
|
|
|
1, // no labels expected
|
|
|
|
},
|
|
|
|
{
|
|
|
|
&Body{
|
|
|
|
Blocks: Blocks{
|
|
|
|
&Block{
|
|
|
|
Type: "foo",
|
|
|
|
Labels: []string{"bar", "baz"},
|
|
|
|
|
|
|
|
LabelRanges: []zcl.Range{{}, {}},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&zcl.BodySchema{
|
|
|
|
Blocks: []zcl.BlockHeaderSchema{
|
|
|
|
{
|
|
|
|
Type: "foo",
|
|
|
|
LabelNames: []string{"name"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
&zcl.BodyContent{
|
|
|
|
Attributes: zcl.Attributes{},
|
|
|
|
},
|
|
|
|
1, // too many labels
|
|
|
|
},
|
|
|
|
{
|
|
|
|
&Body{
|
|
|
|
Attributes: Attributes{
|
|
|
|
"foo": &Attribute{
|
|
|
|
Name: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&zcl.BodySchema{
|
|
|
|
Blocks: []zcl.BlockHeaderSchema{
|
|
|
|
{
|
|
|
|
Type: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
&zcl.BodyContent{
|
|
|
|
Attributes: zcl.Attributes{},
|
|
|
|
},
|
|
|
|
1, // should've been a block, not an attribute
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
prettyConfig := &pretty.Config{
|
|
|
|
Diffable: true,
|
|
|
|
IncludeUnexported: true,
|
|
|
|
PrintStringers: true,
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, test := range tests {
|
|
|
|
t.Run(fmt.Sprintf("%02d", i), func(t *testing.T) {
|
|
|
|
var got *zcl.BodyContent
|
|
|
|
var diags zcl.Diagnostics
|
|
|
|
if test.partial {
|
|
|
|
got, _, diags = test.body.PartialContent(test.schema)
|
|
|
|
} else {
|
|
|
|
got, diags = test.body.Content(test.schema)
|
|
|
|
}
|
|
|
|
|
|
|
|
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 !reflect.DeepEqual(got, test.want) {
|
|
|
|
t.Errorf(
|
|
|
|
"wrong result\ndiff: %s",
|
|
|
|
prettyConfig.Compare(test.want, got),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-03 14:29:48 +00:00
|
|
|
func TestBodyJustAttributes(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
body *Body
|
|
|
|
want zcl.Attributes
|
|
|
|
diagCount int
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
&Body{},
|
|
|
|
zcl.Attributes{},
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
&Body{
|
|
|
|
Attributes: Attributes{},
|
|
|
|
},
|
|
|
|
zcl.Attributes{},
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
&Body{
|
|
|
|
Attributes: Attributes{
|
|
|
|
"foo": &Attribute{
|
|
|
|
Name: "foo",
|
|
|
|
Expr: &LiteralValueExpr{
|
|
|
|
Val: cty.StringVal("bar"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
zcl.Attributes{
|
|
|
|
"foo": &zcl.Attribute{
|
|
|
|
Name: "foo",
|
|
|
|
Expr: &LiteralValueExpr{
|
|
|
|
Val: cty.StringVal("bar"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
&Body{
|
|
|
|
Attributes: Attributes{
|
|
|
|
"foo": &Attribute{
|
|
|
|
Name: "foo",
|
|
|
|
Expr: &LiteralValueExpr{
|
|
|
|
Val: cty.StringVal("bar"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Blocks: Blocks{
|
|
|
|
{
|
|
|
|
Type: "foo",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
zcl.Attributes{
|
|
|
|
"foo": &zcl.Attribute{
|
|
|
|
Name: "foo",
|
|
|
|
Expr: &LiteralValueExpr{
|
|
|
|
Val: cty.StringVal("bar"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
1, // blocks are not allowed here
|
|
|
|
},
|
|
|
|
{
|
|
|
|
&Body{
|
|
|
|
Attributes: Attributes{
|
|
|
|
"foo": &Attribute{
|
|
|
|
Name: "foo",
|
|
|
|
Expr: &LiteralValueExpr{
|
|
|
|
Val: cty.StringVal("bar"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
hiddenAttrs: map[string]struct{}{
|
|
|
|
"foo": struct{}{},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
zcl.Attributes{},
|
|
|
|
0,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
prettyConfig := &pretty.Config{
|
|
|
|
Diffable: true,
|
|
|
|
IncludeUnexported: true,
|
|
|
|
PrintStringers: true,
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, test := range tests {
|
|
|
|
t.Run(fmt.Sprintf("%02d", i), func(t *testing.T) {
|
|
|
|
got, diags := test.body.JustAttributes()
|
|
|
|
|
|
|
|
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 !reflect.DeepEqual(got, test.want) {
|
|
|
|
t.Errorf(
|
|
|
|
"wrong result\nbody: %s\ndiff: %s",
|
|
|
|
prettyConfig.Sprint(test.body),
|
|
|
|
prettyConfig.Compare(test.want, got),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|