2015-10-07 22:38:39 +00:00
|
|
|
package parser
|
|
|
|
|
|
|
|
import "github.com/fatih/hcl/scanner"
|
|
|
|
|
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-09 09:36:40 +00:00
|
|
|
Pos() scanner.Pos
|
2015-10-07 22:38:39 +00:00
|
|
|
}
|
|
|
|
|
2015-10-15 22:11:54 +00:00
|
|
|
func (ObjectList) node() {}
|
|
|
|
func (ObjectItem) node() {}
|
|
|
|
func (ObjectKey) node() {}
|
|
|
|
|
2015-10-12 19:53:40 +00:00
|
|
|
func (ObjectType) node() {}
|
2015-10-15 21:57:57 +00:00
|
|
|
func (LiteralType) node() {}
|
2015-10-12 19:53:40 +00:00
|
|
|
func (ListType) node() {}
|
|
|
|
|
2015-10-15 21:57:57 +00:00
|
|
|
// ObjectList represents a list of ObjectItems. An HCL file itself is an
|
|
|
|
// ObjectList.
|
|
|
|
type ObjectList struct {
|
|
|
|
items []*ObjectItem
|
2015-10-11 22:38:59 +00:00
|
|
|
}
|
|
|
|
|
2015-10-15 21:57:57 +00:00
|
|
|
func (o *ObjectList) add(item *ObjectItem) {
|
|
|
|
o.items = append(o.items, item)
|
2015-10-11 22:38:59 +00:00
|
|
|
}
|
|
|
|
|
2015-10-15 21:57:57 +00:00
|
|
|
func (o *ObjectList) Pos() scanner.Pos {
|
2015-10-11 22:38:59 +00:00
|
|
|
// always returns the uninitiliazed position
|
2015-10-15 21:57:57 +00:00
|
|
|
return o.items[0].Pos()
|
2015-10-11 22:38:59 +00:00
|
|
|
}
|
|
|
|
|
2015-10-15 21:57:57 +00:00
|
|
|
// ObjectItem represents a HCL Object Item. An item is represented with a key
|
|
|
|
// (or keys). It can be an assignment or an object (both normal and nested)
|
|
|
|
type ObjectItem struct {
|
2015-10-15 23:00:02 +00:00
|
|
|
// keys is only one length long if it's of type assignment. If it's a
|
2015-10-15 22:11:54 +00:00
|
|
|
// nested object it can be larger than one. In that case "assign" is
|
|
|
|
// invalid as there is no assignments for a nested object.
|
|
|
|
keys []*ObjectKey
|
2015-10-11 21:20:17 +00:00
|
|
|
|
2015-10-15 21:57:57 +00:00
|
|
|
// assign contains the position of "=", if any
|
|
|
|
assign scanner.Pos
|
2015-10-07 22:38:39 +00:00
|
|
|
|
2015-10-15 21:57:57 +00:00
|
|
|
// val is the item itself. It can be an object,list, number, bool or a
|
2015-10-15 23:00:02 +00:00
|
|
|
// string. If key length is larger than one, val can be only of type
|
2015-10-15 21:57:57 +00:00
|
|
|
// Object.
|
|
|
|
val Node
|
2015-10-11 21:20:17 +00:00
|
|
|
}
|
|
|
|
|
2015-10-15 21:57:57 +00:00
|
|
|
func (o *ObjectItem) Pos() scanner.Pos {
|
2015-10-15 22:11:54 +00:00
|
|
|
return o.keys[0].Pos()
|
2015-10-07 22:38:39 +00:00
|
|
|
}
|
|
|
|
|
2015-10-15 22:11:54 +00:00
|
|
|
// ObjectKeys are either an identifier or of type string.
|
|
|
|
type ObjectKey struct {
|
2015-10-15 21:57:57 +00:00
|
|
|
token scanner.Token
|
2015-10-12 19:53:40 +00:00
|
|
|
}
|
|
|
|
|
2015-10-15 22:11:54 +00:00
|
|
|
func (o *ObjectKey) Pos() scanner.Pos {
|
|
|
|
return o.token.Pos
|
|
|
|
}
|
|
|
|
|
2015-10-15 23:00:02 +00:00
|
|
|
// isValid() returns true if the underlying identifier satisfies one of the
|
|
|
|
// valid types (IDENT or STRING)
|
|
|
|
func (o *ObjectKey) isValid() bool {
|
2015-10-15 22:11:54 +00:00
|
|
|
switch o.token.Type {
|
|
|
|
case scanner.IDENT, scanner.STRING:
|
|
|
|
return true
|
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
2015-10-12 19:53:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// LiteralType represents a literal of basic type. Valid types are:
|
|
|
|
// scanner.NUMBER, scanner.FLOAT, scanner.BOOL and scanner.STRING
|
|
|
|
type LiteralType struct {
|
2015-10-14 22:27:35 +00:00
|
|
|
token scanner.Token
|
2015-10-12 19:53:40 +00:00
|
|
|
}
|
|
|
|
|
2015-10-12 20:44:53 +00:00
|
|
|
// isValid() returns true if the underlying identifier satisfies one of the
|
|
|
|
// valid types.
|
|
|
|
func (l *LiteralType) isValid() bool {
|
|
|
|
switch l.token.Type {
|
|
|
|
case scanner.NUMBER, scanner.FLOAT, scanner.BOOL, scanner.STRING:
|
|
|
|
return true
|
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
2015-10-12 19:53:40 +00:00
|
|
|
}
|
|
|
|
|
2015-10-14 22:27:35 +00:00
|
|
|
func (l *LiteralType) Pos() scanner.Pos {
|
|
|
|
return l.token.Pos
|
|
|
|
}
|
|
|
|
|
2015-10-12 19:53:40 +00:00
|
|
|
// 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) 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) Pos() scanner.Pos {
|
|
|
|
return b.lbrace
|
2015-10-11 21:20:17 +00:00
|
|
|
}
|