hcl/ast/ast.go
2014-08-03 17:17:17 -07:00

121 lines
2.2 KiB
Go

package ast
// ValueType is an enum represnting the type of a value in
// a LiteralNode.
type ValueType byte
const (
ValueTypeUnknown ValueType = iota
ValueTypeFloat
ValueTypeInt
ValueTypeString
ValueTypeBool
ValueTypeNil
)
// Node is implemented by all AST nodes for HCL.
type Node interface {
Accept(Visitor)
}
// KeyedNode is a node that has a key associated with it.
type KeyedNode interface {
Node
Key() string
}
// Visitor is the interface that must be implemented by any
// structures who want to be visited as part of the visitor pattern
// on the AST.
type Visitor interface {
Visit(Node)
}
// ObjectNode represents an object that has multiple elements.
// An object's elements may repeat (keys). This is expected to
// be validated/removed at a semantic check, rather than at a
// syntax level.
type ObjectNode struct {
K string
Elem []KeyedNode
}
// AssignmentNode represents a direct assignment with an equals
// sign.
type AssignmentNode struct {
K string
Value Node
}
// ListNode represents a list or array of items.
type ListNode struct {
Elem []Node
}
// LiteralNode is a direct value.
type LiteralNode struct {
Type ValueType
Value interface{}
}
func (n ObjectNode) Accept(v Visitor) {
v.Visit(n)
for _, e := range n.Elem {
e.Accept(v)
}
}
// Get returns all the elements of this object with the given key.
// This is a case-sensitive search.
func (n ObjectNode) Get(k string) []Node {
result := make([]Node, 0, 1)
for _, elem := range n.Elem {
if elem.Key() != k {
continue
}
switch n := elem.(type) {
case AssignmentNode:
result = append(result, n.Value)
default:
panic("unknown type")
}
}
return result
}
// Key returns the key of this object. If this is "", then it is
// the root object.
func (n ObjectNode) Key() string {
return n.K
}
// Len returns the number of elements of this object.
func (n ObjectNode) Len() int {
return len(n.Elem)
}
func (n AssignmentNode) Accept(v Visitor) {
v.Visit(n)
n.Value.Accept(v)
}
func (n AssignmentNode) Key() string {
return n.K
}
func (n ListNode) Accept(v Visitor) {
v.Visit(n)
for _, e := range n.Elem {
e.Accept(v)
}
}
func (n LiteralNode) Accept(v Visitor) {
v.Visit(n)
}