diff --git a/ast/ast.go b/ast/ast.go index e4370e9..df85e13 100644 --- a/ast/ast.go +++ b/ast/ast.go @@ -69,11 +69,18 @@ func (n ObjectNode) Accept(v Visitor) { // Get returns all the elements of this object with the given key. // This is a case-sensitive search. -func (n ObjectNode) Get(k string) []KeyedNode { - result := make([]KeyedNode, 0, 1) +func (n ObjectNode) Get(k string) []Node { + result := make([]Node, 0, 1) for _, elem := range n.Elem { - if elem.Key() == k { - result = append(result, elem) + if elem.Key() != k { + continue + } + + switch n := elem.(type) { + case AssignmentNode: + result = append(result, n.Value) + default: + panic("unknown type") } } diff --git a/ast/ast_test.go b/ast/ast_test.go index 7519cee..fa45382 100644 --- a/ast/ast_test.go +++ b/ast/ast_test.go @@ -70,3 +70,25 @@ func TestObjectNode_accept(t *testing.T) { t.Fatalf("bad: %#v", v.Nodes) } } + +func TestObjectNodeGet(t *testing.T) { + n := ObjectNode{ + K: "foo", + Elem: []KeyedNode{ + AssignmentNode{K: "foo", Value: LiteralNode{Value: "foo"}}, + AssignmentNode{K: "bar", Value: LiteralNode{Value: "bar"}}, + AssignmentNode{K: "foo", Value: LiteralNode{Value: "baz"}}, + }, + } + + expected := []Node{ + LiteralNode{Value: "foo"}, + LiteralNode{Value: "baz"}, + } + + actual := n.Get("foo") + + if !reflect.DeepEqual(actual, expected) { + t.Fatalf("bad: %#v", actual) + } +} diff --git a/decoder.go b/decoder.go index b734503..48ec3e9 100644 --- a/decoder.go +++ b/decoder.go @@ -58,6 +58,8 @@ func (d *decoder) decode(name string, n ast.Node, result reflect.Value) error { return d.decodeInterface(name, n, result) case reflect.Map: return d.decodeMap(name, n, result) + case reflect.Ptr: + return d.decodePtr(name, n, result) case reflect.Slice: return d.decodeSlice(name, n, result) case reflect.String: @@ -238,6 +240,20 @@ func (d *decoder) decodeMap(name string, raw ast.Node, result reflect.Value) err return nil } +func (d *decoder) decodePtr(name string, raw ast.Node, result reflect.Value) error { + // Create an element of the concrete (non pointer) type and decode + // into that. Then set the value of the pointer to this type. + resultType := result.Type() + resultElemType := resultType.Elem() + val := reflect.New(resultElemType) + if err := d.decode(name, raw, reflect.Indirect(val)); err != nil { + return err + } + + result.Set(val) + return nil +} + func (d *decoder) decodeSlice(name string, raw ast.Node, result reflect.Value) error { n, ok := raw.(ast.ListNode) if !ok { diff --git a/decoder_test.go b/decoder_test.go index 7f23637..6f4095c 100644 --- a/decoder_test.go +++ b/decoder_test.go @@ -160,3 +160,26 @@ func TestDecode_structure(t *testing.T) { t.Fatalf("Actual: %#v\n\nExpected: %#v", actual, expected) } } + +func TestDecode_structurePtr(t *testing.T) { + type V struct { + Key int + Foo string + } + + var actual *V + + err := Decode(&actual, testReadFile(t, "flat.hcl")) + if err != nil { + t.Fatalf("err: %s", err) + } + + expected := &V{ + Key: 7, + Foo: "bar", + } + + if !reflect.DeepEqual(actual, expected) { + t.Fatalf("Actual: %#v\n\nExpected: %#v", actual, expected) + } +}