2015-10-07 22:38:39 +00:00
|
|
|
package parser
|
|
|
|
|
|
|
|
import "github.com/fatih/hcl/scanner"
|
|
|
|
|
|
|
|
type NodeType int
|
|
|
|
|
|
|
|
const (
|
|
|
|
Unknown NodeType = 0
|
|
|
|
Number
|
|
|
|
Float
|
|
|
|
Bool
|
|
|
|
String
|
|
|
|
List
|
|
|
|
Object
|
|
|
|
)
|
|
|
|
|
2015-10-11 22:38:59 +00:00
|
|
|
// Node is an element in the abstract syntax tree.
|
2015-10-07 22:38:39 +00:00
|
|
|
type Node interface {
|
2015-10-11 21:20:17 +00:00
|
|
|
node()
|
2015-10-07 22:38:39 +00:00
|
|
|
String() string
|
2015-10-09 09:36:40 +00:00
|
|
|
Pos() scanner.Pos
|
2015-10-07 22:38:39 +00:00
|
|
|
}
|
|
|
|
|
2015-10-12 19:53:40 +00:00
|
|
|
func (Source) node() {}
|
|
|
|
func (Ident) node() {}
|
|
|
|
|
2015-10-11 21:20:17 +00:00
|
|
|
func (AssignStatement) node() {}
|
|
|
|
func (ObjectStatement) node() {}
|
|
|
|
|
2015-10-12 19:53:40 +00:00
|
|
|
func (LiteralType) node() {}
|
|
|
|
func (ObjectType) node() {}
|
|
|
|
func (ListType) node() {}
|
|
|
|
|
2015-10-11 22:38:59 +00:00
|
|
|
// Source represents a single HCL source file
|
|
|
|
type Source struct {
|
|
|
|
nodes []Node
|
|
|
|
}
|
|
|
|
|
2015-10-11 23:27:43 +00:00
|
|
|
func (s *Source) add(node Node) {
|
2015-10-11 22:38:59 +00:00
|
|
|
s.nodes = append(s.nodes, node)
|
|
|
|
}
|
|
|
|
|
2015-10-11 23:27:43 +00:00
|
|
|
func (s *Source) String() string {
|
2015-10-11 22:38:59 +00:00
|
|
|
buf := ""
|
|
|
|
for _, n := range s.nodes {
|
|
|
|
buf += n.String()
|
|
|
|
}
|
|
|
|
|
|
|
|
return buf
|
|
|
|
}
|
|
|
|
|
2015-10-11 23:27:43 +00:00
|
|
|
func (s *Source) Pos() scanner.Pos {
|
2015-10-11 22:38:59 +00:00
|
|
|
// always returns the uninitiliazed position
|
2015-10-11 23:49:07 +00:00
|
|
|
return s.nodes[0].Pos()
|
2015-10-11 22:38:59 +00:00
|
|
|
}
|
|
|
|
|
2015-10-07 22:38:39 +00:00
|
|
|
// IdentStatement represents an identifier.
|
2015-10-11 23:28:27 +00:00
|
|
|
type Ident struct {
|
2015-10-11 21:20:17 +00:00
|
|
|
token scanner.Token
|
|
|
|
}
|
|
|
|
|
2015-10-11 23:28:27 +00:00
|
|
|
func (i *Ident) String() string {
|
2015-10-12 19:53:40 +00:00
|
|
|
return i.token.Text
|
2015-10-11 21:20:17 +00:00
|
|
|
}
|
|
|
|
|
2015-10-11 23:28:27 +00:00
|
|
|
func (i *Ident) Pos() scanner.Pos {
|
2015-10-11 23:27:43 +00:00
|
|
|
return i.token.Pos
|
2015-10-07 22:38:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// AssignStatement represents an assignment
|
|
|
|
type AssignStatement struct {
|
2015-10-11 21:20:17 +00:00
|
|
|
lhs Node // left hand side of the assignment
|
|
|
|
rhs Node // right hand side of the assignment
|
|
|
|
assign scanner.Pos // position of "="
|
|
|
|
}
|
|
|
|
|
2015-10-11 23:27:43 +00:00
|
|
|
func (a *AssignStatement) String() string {
|
2015-10-11 21:20:17 +00:00
|
|
|
return a.lhs.String() + " = " + a.rhs.String()
|
|
|
|
}
|
|
|
|
|
2015-10-11 23:27:43 +00:00
|
|
|
func (a *AssignStatement) Pos() scanner.Pos {
|
2015-10-11 21:20:17 +00:00
|
|
|
return a.lhs.Pos()
|
2015-10-07 22:38:39 +00:00
|
|
|
}
|
|
|
|
|
2015-10-12 19:53:40 +00:00
|
|
|
// ObjectStatment represents an object statement
|
|
|
|
type ObjectStatement struct {
|
|
|
|
Idents []Node // the idents in elements in lexical order
|
|
|
|
ObjectType
|
|
|
|
}
|
|
|
|
|
|
|
|
func (o *ObjectStatement) String() string {
|
|
|
|
s := ""
|
|
|
|
|
|
|
|
for i, n := range o.Idents {
|
|
|
|
s += n.String()
|
|
|
|
if i != len(o.Idents) {
|
|
|
|
s += " "
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
s += o.ObjectType.String()
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
|
|
|
func (o *ObjectStatement) Pos() scanner.Pos {
|
|
|
|
return o.Idents[0].Pos()
|
|
|
|
}
|
|
|
|
|
|
|
|
// LiteralType represents a literal of basic type. Valid types are:
|
|
|
|
// scanner.NUMBER, scanner.FLOAT, scanner.BOOL and scanner.STRING
|
|
|
|
type LiteralType struct {
|
|
|
|
token scanner.Token
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *LiteralType) String() string {
|
|
|
|
return l.token.Text
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *LiteralType) Pos() scanner.Pos {
|
|
|
|
return l.token.Pos
|
|
|
|
}
|
|
|
|
|
|
|
|
// ListStatement represents a HCL List type
|
|
|
|
type ListType struct {
|
2015-10-11 21:20:17 +00:00
|
|
|
lbrack scanner.Pos // position of "["
|
|
|
|
rbrack scanner.Pos // position of "]"
|
|
|
|
list []Node // the elements in lexical order
|
|
|
|
}
|
|
|
|
|
2015-10-12 19:53:40 +00:00
|
|
|
func (l *ListType) String() string {
|
2015-10-11 21:20:17 +00:00
|
|
|
s := "[\n"
|
|
|
|
for _, n := range l.list {
|
|
|
|
s += n.String() + ",\n"
|
|
|
|
}
|
|
|
|
|
|
|
|
s += "]"
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
2015-10-12 19:53:40 +00:00
|
|
|
func (l *ListType) Pos() scanner.Pos {
|
2015-10-11 21:20:17 +00:00
|
|
|
return l.lbrack
|
2015-10-07 22:38:39 +00:00
|
|
|
}
|
|
|
|
|
2015-10-12 19:53:40 +00:00
|
|
|
// ObjectType represents a HCL Object Type
|
|
|
|
type ObjectType struct {
|
|
|
|
lbrace scanner.Pos // position of "{"
|
|
|
|
rbrace scanner.Pos // position of "}"
|
|
|
|
list []Node // the nodes in lexical order
|
2015-10-07 22:38:39 +00:00
|
|
|
}
|
2015-10-11 21:20:17 +00:00
|
|
|
|
2015-10-12 19:53:40 +00:00
|
|
|
func (b *ObjectType) String() string {
|
|
|
|
s := "{\n"
|
|
|
|
for _, n := range b.list {
|
|
|
|
s += n.String() + "\n"
|
2015-10-11 21:20:17 +00:00
|
|
|
}
|
|
|
|
|
2015-10-12 19:53:40 +00:00
|
|
|
s += "}"
|
2015-10-11 21:20:17 +00:00
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
2015-10-12 19:53:40 +00:00
|
|
|
func (b *ObjectType) Pos() scanner.Pos {
|
|
|
|
return b.lbrace
|
2015-10-11 21:20:17 +00:00
|
|
|
}
|