2017-09-11 23:00:31 +00:00
|
|
|
package hclwrite
|
2017-06-07 14:06:23 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"testing"
|
|
|
|
|
2018-08-01 15:45:22 +00:00
|
|
|
"github.com/sergi/go-diff/diffmatchpatch"
|
2017-06-10 21:14:01 +00:00
|
|
|
"github.com/zclconf/go-cty/cty"
|
|
|
|
"github.com/zclconf/go-cty/cty/function"
|
|
|
|
"github.com/zclconf/go-cty/cty/function/stdlib"
|
2018-08-01 15:45:22 +00:00
|
|
|
|
|
|
|
"github.com/hashicorp/hcl2/hcl"
|
|
|
|
"github.com/hashicorp/hcl2/hcl/hclsyntax"
|
2017-06-07 14:06:23 +00:00
|
|
|
)
|
|
|
|
|
2017-06-10 21:14:01 +00:00
|
|
|
func TestRoundTripVerbatim(t *testing.T) {
|
2017-06-07 14:06:23 +00:00
|
|
|
tests := []string{
|
|
|
|
``,
|
2017-06-09 15:19:47 +00:00
|
|
|
`foo = 1
|
|
|
|
`,
|
2017-06-07 14:06:23 +00:00
|
|
|
`
|
|
|
|
foobar = 1
|
|
|
|
baz = 1
|
|
|
|
`,
|
|
|
|
`
|
|
|
|
# this file is awesome
|
|
|
|
|
|
|
|
# tossed salads and scrambled eggs
|
2017-06-08 16:04:27 +00:00
|
|
|
foobar = 1
|
|
|
|
baz = 1
|
2017-06-07 14:06:23 +00:00
|
|
|
|
2017-06-11 00:16:19 +00:00
|
|
|
block {
|
|
|
|
a = "a"
|
|
|
|
b = "b"
|
|
|
|
c = "c"
|
|
|
|
d = "d"
|
|
|
|
|
|
|
|
subblock {
|
|
|
|
}
|
|
|
|
|
|
|
|
subblock {
|
|
|
|
e = "e"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-07 14:06:23 +00:00
|
|
|
# and they all lived happily ever after
|
|
|
|
`,
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range tests {
|
|
|
|
t.Run(test, func(t *testing.T) {
|
|
|
|
src := []byte(test)
|
2017-09-11 23:40:37 +00:00
|
|
|
file, diags := parse(src, "", hcl.Pos{Line: 1, Column: 1})
|
2017-06-07 14:06:23 +00:00
|
|
|
if len(diags) != 0 {
|
|
|
|
for _, diag := range diags {
|
|
|
|
t.Logf(" - %s", diag.Error())
|
|
|
|
}
|
|
|
|
t.Fatalf("unexpected diagnostics")
|
|
|
|
}
|
|
|
|
|
|
|
|
wr := &bytes.Buffer{}
|
|
|
|
n, err := file.WriteTo(wr)
|
2018-11-03 20:10:05 +00:00
|
|
|
if n != int64(len(test)) {
|
2017-06-07 14:06:23 +00:00
|
|
|
t.Errorf("wrong number of bytes %d; want %d", n, len(test))
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("error from WriteTo")
|
|
|
|
}
|
|
|
|
|
|
|
|
result := wr.Bytes()
|
|
|
|
|
|
|
|
if !bytes.Equal(result, src) {
|
2018-08-01 15:45:22 +00:00
|
|
|
dmp := diffmatchpatch.New()
|
|
|
|
diffs := dmp.DiffMain(string(src), string(result), false)
|
|
|
|
//t.Errorf("wrong result\nresult:\n%s\ninput:\n%s", result, src)
|
|
|
|
t.Errorf("wrong result\ndiff: (red indicates missing lines, and green indicates unexpected lines)\n%s", dmp.DiffPrettyText(diffs))
|
2017-06-07 14:06:23 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2017-06-10 21:14:01 +00:00
|
|
|
|
|
|
|
func TestRoundTripFormat(t *testing.T) {
|
|
|
|
// The goal of this test is to verify that the formatter doesn't change
|
|
|
|
// the semantics of any expressions when it adds and removes whitespace.
|
|
|
|
// String templates are the primary area of concern here, but we also
|
|
|
|
// test some other things for completeness sake.
|
|
|
|
//
|
|
|
|
// The tests here must define zero or more attributes, which will be
|
|
|
|
// extract with JustAttributes and evaluated both before and after
|
|
|
|
// formatting.
|
|
|
|
|
|
|
|
tests := []string{
|
|
|
|
"",
|
|
|
|
"\n\n\n",
|
|
|
|
"a=1\n",
|
|
|
|
"a=\"hello\"\n",
|
|
|
|
"a=\"${hello} world\"\n",
|
|
|
|
"a=upper(\"hello\")\n",
|
|
|
|
"a=upper(hello)\n",
|
|
|
|
"a=[1,2,3,4,five]\n",
|
|
|
|
"a={greeting=hello}\n",
|
|
|
|
"a={\ngreeting=hello\n}\n",
|
|
|
|
"a={\ngreeting=hello}\n",
|
|
|
|
"a={greeting=hello\n}\n",
|
|
|
|
"a={greeting=hello,number=five,sarcastic=\"${upper(hello)}\"\n}\n",
|
|
|
|
"a={\ngreeting=hello\nnumber=five\nsarcastic=\"${upper(hello)}\"\n}\n",
|
|
|
|
"a=<<EOT\nhello\nEOT\n\n",
|
|
|
|
"a=[<<EOT\nhello\nEOT\n]\n",
|
|
|
|
"a=[\n<<EOT\nhello\nEOT\n]\n",
|
|
|
|
"a=[\n]\n",
|
|
|
|
"a=1\nb=2\nc=3\n",
|
|
|
|
"a=\"${\n5\n}\"\n",
|
|
|
|
}
|
|
|
|
|
2017-09-11 23:40:37 +00:00
|
|
|
ctx := &hcl.EvalContext{
|
2017-06-10 21:14:01 +00:00
|
|
|
Variables: map[string]cty.Value{
|
|
|
|
"hello": cty.StringVal("hello"),
|
|
|
|
"five": cty.NumberIntVal(5),
|
|
|
|
},
|
|
|
|
Functions: map[string]function.Function{
|
|
|
|
"upper": stdlib.UpperFunc,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range tests {
|
|
|
|
t.Run(test, func(t *testing.T) {
|
|
|
|
|
|
|
|
attrsAsObj := func(src []byte, phase string) cty.Value {
|
|
|
|
t.Logf("source %s:\n%s", phase, src)
|
2017-09-11 23:40:37 +00:00
|
|
|
f, diags := hclsyntax.ParseConfig(src, "", hcl.Pos{Line: 1, Column: 1})
|
2017-06-10 21:14:01 +00:00
|
|
|
if len(diags) != 0 {
|
|
|
|
for _, diag := range diags {
|
|
|
|
t.Logf(" - %s", diag.Error())
|
|
|
|
}
|
|
|
|
t.Fatalf("unexpected diagnostics in parse %s", phase)
|
|
|
|
}
|
|
|
|
|
|
|
|
attrs, diags := f.Body.JustAttributes()
|
|
|
|
if len(diags) != 0 {
|
|
|
|
for _, diag := range diags {
|
|
|
|
t.Logf(" - %s", diag.Error())
|
|
|
|
}
|
|
|
|
t.Fatalf("unexpected diagnostics in JustAttributes %s", phase)
|
|
|
|
}
|
|
|
|
|
|
|
|
vals := map[string]cty.Value{}
|
|
|
|
for k, attr := range attrs {
|
|
|
|
val, diags := attr.Expr.Value(ctx)
|
|
|
|
if len(diags) != 0 {
|
|
|
|
for _, diag := range diags {
|
|
|
|
t.Logf(" - %s", diag.Error())
|
|
|
|
}
|
|
|
|
t.Fatalf("unexpected diagnostics evaluating %s", phase)
|
|
|
|
}
|
|
|
|
vals[k] = val
|
|
|
|
}
|
|
|
|
return cty.ObjectVal(vals)
|
|
|
|
}
|
|
|
|
|
|
|
|
src := []byte(test)
|
|
|
|
before := attrsAsObj(src, "before")
|
|
|
|
|
|
|
|
formatted := Format(src)
|
|
|
|
after := attrsAsObj(formatted, "after")
|
|
|
|
|
|
|
|
if !after.RawEquals(before) {
|
|
|
|
t.Errorf("mismatching after format\nbefore: %#v\nafter: %#v", before, after)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|