hcl/parser/parser.go

233 lines
4.8 KiB
Go
Raw Normal View History

package parser
2015-10-07 22:38:39 +00:00
2015-10-11 23:27:43 +00:00
import (
2015-10-12 19:53:40 +00:00
"errors"
2015-10-11 23:27:43 +00:00
"fmt"
"github.com/fatih/hcl/scanner"
)
2015-10-07 22:38:39 +00:00
type Parser struct {
sc *scanner.Scanner
tok scanner.Token // last read token
prevTok scanner.Token // previous read token
2015-10-11 23:27:43 +00:00
enableTrace bool
indent int
n int // buffer size (max = 1)
2015-10-07 22:38:39 +00:00
}
2015-10-11 23:27:43 +00:00
func New(src []byte) *Parser {
2015-10-07 22:38:39 +00:00
return &Parser{
sc: scanner.New(src),
}
}
var errEofToken = errors.New("EOF token found")
// Parse returns the fully parsed source and returns the abstract syntax tree.
2015-10-12 07:37:37 +00:00
func (p *Parser) Parse() (Node, error) {
2015-10-11 23:27:43 +00:00
defer un(trace(p, "ParseSource"))
node := &ObjectList{}
2015-10-11 23:27:43 +00:00
for {
n, err := p.parseObjectItem()
if err == errEofToken {
break // we are finished
}
2015-10-12 19:53:40 +00:00
if err != nil {
return nil, err
}
// we successfully parsed a node, add it to the final source node
2015-10-12 19:53:40 +00:00
node.add(n)
2015-10-07 22:38:39 +00:00
}
2015-10-12 07:37:37 +00:00
return node, nil
}
// parseObjectItem parses a single object item
func (p *Parser) parseObjectItem() (*ObjectItem, error) {
defer un(trace(p, "ParseObjectItem"))
2015-10-11 23:27:43 +00:00
keys, err := p.parseObjectKey()
if err != nil {
return nil, err
}
2015-10-16 11:16:12 +00:00
// either an assignment or object
switch p.tok.Type {
case scanner.ASSIGN:
2015-10-16 11:44:11 +00:00
o := &ObjectItem{
keys: keys,
assign: p.tok.Pos,
}
o.val, err = p.parseType()
if err != nil {
return nil, err
}
return o, nil
2015-10-16 11:16:12 +00:00
case scanner.LBRACE:
if len(keys) > 1 {
// nested object
2015-10-16 11:44:11 +00:00
fmt.Println("nested object")
2015-10-16 11:16:12 +00:00
}
2015-10-16 11:44:11 +00:00
// object
fmt.Println("object")
2015-10-16 11:16:12 +00:00
}
switch len(keys) {
case 1:
// assignment or object
default:
// nested object
}
tok := p.scan()
2015-10-12 19:53:40 +00:00
fmt.Println(tok) // debug
switch tok.Type {
case scanner.LBRACK:
// return p.parseListType()
case scanner.LBRACE:
// return p.parseObjectTpe()
case scanner.COMMENT:
// implement comment
case scanner.EOF:
return nil, errEofToken
}
return nil, fmt.Errorf("not yet implemented: %s", tok.Type)
}
2015-10-16 11:44:11 +00:00
// parseType parses any type of Type, such as number, bool, string, object or
// list.
func (p *Parser) parseType() (Node, error) {
return nil, errors.New("ParseType is not implemented yet")
}
// parseObjectKey parses an object key and returns a ObjectKey AST
func (p *Parser) parseObjectKey() ([]*ObjectKey, error) {
tok := p.scan()
2015-10-16 11:16:12 +00:00
keys := make([]*ObjectKey, 0)
switch tok.Type {
case scanner.IDENT, scanner.STRING:
// add first found token
2015-10-16 11:16:12 +00:00
keys = append(keys, &ObjectKey{token: tok})
default:
return nil, fmt.Errorf("expected: IDENT | STRING got: %s", tok.Type)
}
2015-10-16 11:16:12 +00:00
nestedObj := false
// we have three casses
// 1. assignment: KEY = NODE
// 2. object: KEY { }
// 2. nested object: KEY KEY2 ... KEYN {}
for {
tok := p.scan()
switch tok.Type {
case scanner.ASSIGN:
2015-10-16 11:44:11 +00:00
// assignment or object only, but not nested objects. this is not
// allowed: `foo bar = {}`
2015-10-16 11:16:12 +00:00
if nestedObj {
return nil, fmt.Errorf("nested object expected: LBRACE got: %s", tok.Type)
}
return keys, nil
case scanner.LBRACE:
// object
return keys, nil
case scanner.IDENT, scanner.STRING:
// nested object
nestedObj = true
keys = append(keys, &ObjectKey{token: tok})
default:
return nil, fmt.Errorf("expected: IDENT | STRING | ASSIGN | LBRACE got: %s", tok.Type)
}
}
}
2015-10-12 20:44:53 +00:00
// parseLiteralType parses a literal type and returns a LiteralType AST
func (p *Parser) parseLiteralType() (*LiteralType, error) {
defer un(trace(p, "ParseLiteral"))
2015-10-12 20:44:53 +00:00
return &LiteralType{
token: p.tok,
}, nil
2015-10-12 20:44:53 +00:00
}
// parseObjectType parses an object type and returns a ObjectType AST
func (p *Parser) parseObjectType() (*ObjectType, error) {
return nil, errors.New("ObjectType is not implemented yet")
}
// parseListType parses a list type and returns a ListType AST
2015-10-12 19:53:40 +00:00
func (p *Parser) parseListType() (*ListType, error) {
2015-10-12 20:44:53 +00:00
return nil, errors.New("ListType is not implemented yet")
}
// scan returns the next token from the underlying scanner.
// If a token has been unscanned then read that instead.
func (p *Parser) scan() scanner.Token {
// If we have a token on the buffer, then return it.
if p.n != 0 {
p.n = 0
return p.tok
}
// store previous token
p.prevTok = p.tok
// Otherwise read the next token from the scanner and Save it to the buffer
// in case we unscan later.
p.tok = p.sc.Scan()
return p.tok
2015-10-07 22:38:39 +00:00
}
// unscan pushes the previously read token back onto the buffer.
func (p *Parser) unscan() {
p.n = 1
p.tok = p.prevTok
}
2015-10-11 23:27:43 +00:00
// ----------------------------------------------------------------------------
// Parsing support
func (p *Parser) printTrace(a ...interface{}) {
if !p.enableTrace {
return
}
const dots = ". . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . "
const n = len(dots)
fmt.Printf("%5d:%3d: ", p.tok.Pos.Line, p.tok.Pos.Column)
i := 2 * p.indent
for i > n {
fmt.Print(dots)
i -= n
}
// i <= n
fmt.Print(dots[0:i])
fmt.Println(a...)
}
func trace(p *Parser, msg string) *Parser {
p.printTrace(msg, "(")
p.indent++
return p
}
// Usage pattern: defer un(trace(p, "..."))
func un(p *Parser) {
p.indent--
p.printTrace(")")
}