lf/parse.go
2016-12-18 00:47:37 +03:00

241 lines
3.7 KiB
Go

package main
// Grammar of the language used in the evaluator
//
// Expr = SetExpr
// | MapExpr
// | CmdExpr
// | CallExpr
// | ExecExpr
// | ListExpr
//
// SetExpr = 'set' <opt> <val> ';'
//
// MapExpr = 'map' <keys> Expr ';'
//
// CmdExpr = 'cmd' <name> Expr ';'
//
// CallExpr = <name> <args> ';'
//
// ExecExpr = Prefix <expr> '\n'
// | Prefix '{{' <expr> '}}' ';'
//
// Prefix = '$' | '!' | '&' | '/' | '?'
//
// ListExpr = ':' ListRest '\n'
// | ':' '{{' ListRest '}}' ';'
//
// ListRest = Nil
// | Expr ListRest
import (
"fmt"
"io"
)
type expr interface {
String() string
eval(app *app, args []string)
// TODO: add a bind method to avoid passing args in eval
}
type setExpr struct {
opt string
val string
}
func (e *setExpr) String() string { return fmt.Sprintf("set %s %s", e.opt, e.val) }
type mapExpr struct {
keys string
expr expr
}
func (e *mapExpr) String() string { return fmt.Sprintf("map %s %s", e.keys, e.expr) }
type cmdExpr struct {
name string
expr expr
}
func (e *cmdExpr) String() string { return fmt.Sprintf("cmd %s %s", e.name, e.expr) }
type callExpr struct {
name string
args []string
}
func (e *callExpr) String() string { return fmt.Sprintf("%s -- %s", e.name, e.args) }
type execExpr struct {
pref string
expr string
}
func (e *execExpr) String() string { return fmt.Sprintf("%s %s", e.pref, e.expr) }
type listExpr struct {
exprs []expr
}
func (e *listExpr) String() string {
buf := []byte{':', '{', '{', ' '}
for _, expr := range e.exprs {
buf = append(buf, expr.String()...)
buf = append(buf, ';', ' ')
}
buf = append(buf, '}', '}')
return string(buf)
}
type parser struct {
scanner *scanner
expr expr
err error
}
func newParser(r io.Reader) *parser {
scanner := newScanner(r)
scanner.scan()
return &parser{
scanner: scanner,
}
}
func (p *parser) parseExpr() expr {
s := p.scanner
var result expr
switch s.typ {
case tokenEOF:
return nil
case tokenIdent:
switch s.tok {
case "set":
var val string
s.scan()
if s.typ != tokenIdent {
p.err = fmt.Errorf("expected identifier: %s", s.tok)
}
opt := s.tok
s.scan()
if s.typ != tokenSemicolon {
val = s.tok
s.scan()
}
s.scan()
result = &setExpr{opt, val}
case "map":
var expr expr
s.scan()
keys := s.tok
s.scan()
if s.typ != tokenSemicolon {
expr = p.parseExpr()
} else {
s.scan()
}
result = &mapExpr{keys, expr}
case "cmd":
var expr expr
s.scan()
name := s.tok
s.scan()
if s.typ != tokenSemicolon {
expr = p.parseExpr()
} else {
s.scan()
}
result = &cmdExpr{name, expr}
default:
name := s.tok
var args []string
for s.scan() && s.typ != tokenSemicolon {
args = append(args, s.tok)
}
s.scan()
result = &callExpr{name, args}
}
case tokenColon:
s.scan()
var exprs []expr
if s.typ == tokenLBraces {
s.scan()
for {
e := p.parseExpr()
if e == nil {
return nil
}
exprs = append(exprs, e)
if s.typ == tokenRBraces {
break
}
}
s.scan()
} else {
for {
e := p.parseExpr()
if e == nil {
return nil
}
exprs = append(exprs, e)
if s.tok == "\n" {
break
}
}
}
s.scan()
result = &listExpr{exprs}
case tokenPrefix:
var expr string
pref := s.tok
s.scan()
if s.typ == tokenLBraces {
s.scan()
expr = s.tok
s.scan()
} else {
expr = s.tok
}
s.scan()
s.scan()
result = &execExpr{pref, expr}
default:
p.err = fmt.Errorf("unexpected token: %s", s.tok)
}
return result
}
func (p *parser) parse() bool {
if p.expr = p.parseExpr(); p.expr == nil {
return false
}
return true
}