package hclwrite import ( "fmt" "reflect" "testing" "github.com/davecgh/go-spew/spew" "github.com/google/go-cmp/cmp" "github.com/hashicorp/hcl/v2" "github.com/hashicorp/hcl/v2/hclsyntax" "github.com/zclconf/go-cty/cty" ) func TestBodyGetAttribute(t *testing.T) { tests := []struct { src string name string want Tokens }{ { "", "a", nil, }, { "a = 1\n", "a", Tokens{ { Type: hclsyntax.TokenIdent, Bytes: []byte{'a'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEqual, Bytes: []byte{'='}, SpacesBefore: 1, }, { Type: hclsyntax.TokenNumberLit, Bytes: []byte{'1'}, SpacesBefore: 1, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, }, }, { "a = 1\nb = 1\nc = 1\n", "a", Tokens{ { Type: hclsyntax.TokenIdent, Bytes: []byte{'a'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEqual, Bytes: []byte{'='}, SpacesBefore: 1, }, { Type: hclsyntax.TokenNumberLit, Bytes: []byte{'1'}, SpacesBefore: 1, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, }, }, { "a = 1\nb = 2\nc = 3\n", "b", Tokens{ { Type: hclsyntax.TokenIdent, Bytes: []byte{'b'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEqual, Bytes: []byte{'='}, SpacesBefore: 1, }, { Type: hclsyntax.TokenNumberLit, Bytes: []byte{'2'}, SpacesBefore: 1, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, }, }, { "a = 1\nb = 2\nc = 3\n", "c", Tokens{ { Type: hclsyntax.TokenIdent, Bytes: []byte{'c'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEqual, Bytes: []byte{'='}, SpacesBefore: 1, }, { Type: hclsyntax.TokenNumberLit, Bytes: []byte{'3'}, SpacesBefore: 1, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, }, }, { "a = 1\n# b is a b\nb = 2\nc = 3\n", "b", Tokens{ { // Recognized as a lead comment and so attached to the attribute Type: hclsyntax.TokenComment, Bytes: []byte("# b is a b\n"), SpacesBefore: 0, }, { Type: hclsyntax.TokenIdent, Bytes: []byte{'b'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEqual, Bytes: []byte{'='}, SpacesBefore: 1, }, { Type: hclsyntax.TokenNumberLit, Bytes: []byte{'2'}, SpacesBefore: 1, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, }, }, { "a = 1\n# not attached to a or b\n\nb = 2\nc = 3\n", "b", Tokens{ { Type: hclsyntax.TokenIdent, Bytes: []byte{'b'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEqual, Bytes: []byte{'='}, SpacesBefore: 1, }, { Type: hclsyntax.TokenNumberLit, Bytes: []byte{'2'}, SpacesBefore: 1, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, }, }, } for _, test := range tests { t.Run(fmt.Sprintf("%s in %s", test.name, test.src), func(t *testing.T) { f, diags := ParseConfig([]byte(test.src), "", hcl.Pos{Line: 1, Column: 1}) if len(diags) != 0 { for _, diag := range diags { t.Logf("- %s", diag.Error()) } t.Fatalf("unexpected diagnostics") } attr := f.Body().GetAttribute(test.name) if attr == nil { if test.want != nil { t.Fatal("attribute not found, but want it to exist") } } else { if test.want == nil { t.Fatal("attribute found, but expecting not found") } got := attr.BuildTokens(nil) if !reflect.DeepEqual(got, test.want) { t.Errorf("wrong result\ngot: %s\nwant: %s", spew.Sdump(got), spew.Sdump(test.want)) } } }) } } func TestBodySetAttributeValue(t *testing.T) { tests := []struct { src string name string val cty.Value want Tokens }{ { "", "a", cty.True, Tokens{ { Type: hclsyntax.TokenIdent, Bytes: []byte{'a'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEqual, Bytes: []byte{'='}, SpacesBefore: 1, }, { Type: hclsyntax.TokenIdent, Bytes: []byte("true"), SpacesBefore: 1, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEOF, Bytes: []byte{}, SpacesBefore: 0, }, }, }, { "b = false\n", "a", cty.True, Tokens{ { Type: hclsyntax.TokenIdent, Bytes: []byte{'b'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEqual, Bytes: []byte{'='}, SpacesBefore: 1, }, { Type: hclsyntax.TokenIdent, Bytes: []byte("false"), SpacesBefore: 1, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenIdent, Bytes: []byte{'a'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEqual, Bytes: []byte{'='}, SpacesBefore: 1, }, { Type: hclsyntax.TokenIdent, Bytes: []byte("true"), SpacesBefore: 1, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEOF, Bytes: []byte{}, SpacesBefore: 0, }, }, }, { "a = false\n", "a", cty.True, Tokens{ { Type: hclsyntax.TokenIdent, Bytes: []byte{'a'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEqual, Bytes: []byte{'='}, SpacesBefore: 1, }, { Type: hclsyntax.TokenIdent, Bytes: []byte("true"), SpacesBefore: 1, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEOF, Bytes: []byte{}, SpacesBefore: 0, }, }, }, { "a = 1\nb = false\n", "a", cty.True, Tokens{ { Type: hclsyntax.TokenIdent, Bytes: []byte{'a'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEqual, Bytes: []byte{'='}, SpacesBefore: 1, }, { Type: hclsyntax.TokenIdent, Bytes: []byte("true"), SpacesBefore: 1, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenIdent, Bytes: []byte{'b'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEqual, Bytes: []byte{'='}, SpacesBefore: 1, }, { Type: hclsyntax.TokenIdent, Bytes: []byte("false"), SpacesBefore: 1, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEOF, Bytes: []byte{}, SpacesBefore: 0, }, }, }, } for _, test := range tests { t.Run(fmt.Sprintf("%s = %#v in %s", test.name, test.val, test.src), func(t *testing.T) { f, diags := ParseConfig([]byte(test.src), "", hcl.Pos{Line: 1, Column: 1}) if len(diags) != 0 { for _, diag := range diags { t.Logf("- %s", diag.Error()) } t.Fatalf("unexpected diagnostics") } f.Body().SetAttributeValue(test.name, test.val) got := f.BuildTokens(nil) format(got) if !reflect.DeepEqual(got, test.want) { diff := cmp.Diff(test.want, got) t.Errorf("wrong result\ngot: %s\nwant: %s\ndiff:\n%s", spew.Sdump(got), spew.Sdump(test.want), diff) } }) } } func TestBodySetAttributeTraversal(t *testing.T) { tests := []struct { src string name string trav string want Tokens }{ { "", "a", `b`, Tokens{ { Type: hclsyntax.TokenIdent, Bytes: []byte{'a'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEqual, Bytes: []byte{'='}, SpacesBefore: 1, }, { Type: hclsyntax.TokenIdent, Bytes: []byte("b"), SpacesBefore: 1, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEOF, Bytes: []byte{}, SpacesBefore: 0, }, }, }, { "", "a", `b.c.d`, Tokens{ { Type: hclsyntax.TokenIdent, Bytes: []byte{'a'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEqual, Bytes: []byte{'='}, SpacesBefore: 1, }, { Type: hclsyntax.TokenIdent, Bytes: []byte("b"), SpacesBefore: 1, }, { Type: hclsyntax.TokenDot, Bytes: []byte("."), SpacesBefore: 0, }, { Type: hclsyntax.TokenIdent, Bytes: []byte("c"), SpacesBefore: 0, }, { Type: hclsyntax.TokenDot, Bytes: []byte("."), SpacesBefore: 0, }, { Type: hclsyntax.TokenIdent, Bytes: []byte("d"), SpacesBefore: 0, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEOF, Bytes: []byte{}, SpacesBefore: 0, }, }, }, { "", "a", `b[0]`, Tokens{ { Type: hclsyntax.TokenIdent, Bytes: []byte{'a'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEqual, Bytes: []byte{'='}, SpacesBefore: 1, }, { Type: hclsyntax.TokenIdent, Bytes: []byte("b"), SpacesBefore: 1, }, { Type: hclsyntax.TokenOBrack, Bytes: []byte("["), SpacesBefore: 0, }, { Type: hclsyntax.TokenNumberLit, Bytes: []byte("0"), SpacesBefore: 0, }, { Type: hclsyntax.TokenCBrack, Bytes: []byte("]"), SpacesBefore: 0, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEOF, Bytes: []byte{}, SpacesBefore: 0, }, }, }, { "", "a", `b[0].c`, Tokens{ { Type: hclsyntax.TokenIdent, Bytes: []byte{'a'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEqual, Bytes: []byte{'='}, SpacesBefore: 1, }, { Type: hclsyntax.TokenIdent, Bytes: []byte("b"), SpacesBefore: 1, }, { Type: hclsyntax.TokenOBrack, Bytes: []byte("["), SpacesBefore: 0, }, { Type: hclsyntax.TokenNumberLit, Bytes: []byte("0"), SpacesBefore: 0, }, { Type: hclsyntax.TokenCBrack, Bytes: []byte("]"), SpacesBefore: 0, }, { Type: hclsyntax.TokenDot, Bytes: []byte("."), SpacesBefore: 0, }, { Type: hclsyntax.TokenIdent, Bytes: []byte("c"), SpacesBefore: 0, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEOF, Bytes: []byte{}, SpacesBefore: 0, }, }, }, } for _, test := range tests { t.Run(fmt.Sprintf("%s = %s in %s", test.name, test.trav, test.src), func(t *testing.T) { f, diags := ParseConfig([]byte(test.src), "", hcl.Pos{Line: 1, Column: 1}) if len(diags) != 0 { for _, diag := range diags { t.Logf("- %s", diag.Error()) } t.Fatalf("unexpected diagnostics") } traversal, diags := hclsyntax.ParseTraversalAbs([]byte(test.trav), "", hcl.Pos{Line: 1, Column: 1}) if len(diags) != 0 { for _, diag := range diags { t.Logf("- %s", diag.Error()) } t.Fatalf("unexpected diagnostics from traversal") } f.Body().SetAttributeTraversal(test.name, traversal) got := f.BuildTokens(nil) format(got) if !reflect.DeepEqual(got, test.want) { diff := cmp.Diff(test.want, got) t.Errorf("wrong result\ngot: %s\nwant: %s\ndiff:\n%s", spew.Sdump(got), spew.Sdump(test.want), diff) } }) } } func TestBodyAppendBlock(t *testing.T) { tests := []struct { src string blockType string labels []string want Tokens }{ { "", "foo", nil, Tokens{ { Type: hclsyntax.TokenIdent, Bytes: []byte(`foo`), SpacesBefore: 0, }, { Type: hclsyntax.TokenOBrace, Bytes: []byte{'{'}, SpacesBefore: 1, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenCBrace, Bytes: []byte{'}'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEOF, Bytes: []byte{}, SpacesBefore: 0, }, }, }, { "", "foo", []string{"bar"}, Tokens{ { Type: hclsyntax.TokenIdent, Bytes: []byte(`foo`), SpacesBefore: 0, }, { Type: hclsyntax.TokenOQuote, Bytes: []byte(`"`), SpacesBefore: 1, }, { Type: hclsyntax.TokenQuotedLit, Bytes: []byte(`bar`), SpacesBefore: 0, }, { Type: hclsyntax.TokenCQuote, Bytes: []byte(`"`), SpacesBefore: 0, }, { Type: hclsyntax.TokenOBrace, Bytes: []byte{'{'}, SpacesBefore: 1, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenCBrace, Bytes: []byte{'}'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEOF, Bytes: []byte{}, SpacesBefore: 0, }, }, }, { "", "foo", []string{"bar", "baz"}, Tokens{ { Type: hclsyntax.TokenIdent, Bytes: []byte(`foo`), SpacesBefore: 0, }, { Type: hclsyntax.TokenOQuote, Bytes: []byte(`"`), SpacesBefore: 1, }, { Type: hclsyntax.TokenQuotedLit, Bytes: []byte(`bar`), SpacesBefore: 0, }, { Type: hclsyntax.TokenCQuote, Bytes: []byte(`"`), SpacesBefore: 0, }, { Type: hclsyntax.TokenOQuote, Bytes: []byte(`"`), SpacesBefore: 1, }, { Type: hclsyntax.TokenQuotedLit, Bytes: []byte(`baz`), SpacesBefore: 0, }, { Type: hclsyntax.TokenCQuote, Bytes: []byte(`"`), SpacesBefore: 0, }, { Type: hclsyntax.TokenOBrace, Bytes: []byte{'{'}, SpacesBefore: 1, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenCBrace, Bytes: []byte{'}'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEOF, Bytes: []byte{}, SpacesBefore: 0, }, }, }, { "bar {}\n", "foo", nil, Tokens{ { Type: hclsyntax.TokenIdent, Bytes: []byte(`bar`), SpacesBefore: 0, }, { Type: hclsyntax.TokenOBrace, Bytes: []byte{'{'}, SpacesBefore: 1, }, { Type: hclsyntax.TokenCBrace, Bytes: []byte{'}'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenIdent, Bytes: []byte(`foo`), SpacesBefore: 0, }, { Type: hclsyntax.TokenOBrace, Bytes: []byte{'{'}, SpacesBefore: 1, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenCBrace, Bytes: []byte{'}'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenNewline, Bytes: []byte{'\n'}, SpacesBefore: 0, }, { Type: hclsyntax.TokenEOF, Bytes: []byte{}, SpacesBefore: 0, }, }, }, } for _, test := range tests { t.Run(fmt.Sprintf("%s %#v in %s", test.blockType, test.blockType, test.src), func(t *testing.T) { f, diags := ParseConfig([]byte(test.src), "", hcl.Pos{Line: 1, Column: 1}) if len(diags) != 0 { for _, diag := range diags { t.Logf("- %s", diag.Error()) } t.Fatalf("unexpected diagnostics") } f.Body().AppendNewBlock(test.blockType, test.labels) got := f.BuildTokens(nil) format(got) if !reflect.DeepEqual(got, test.want) { diff := cmp.Diff(test.want, got) t.Errorf("wrong result\ngot: %s\nwant: %s\ndiff:\n%s", spew.Sdump(got), spew.Sdump(test.want), diff) } }) } }