2024-03-26 08:45:18 +01:00
|
|
|
// Copyright (c) 2024 Celestino Amoroso (celestino.amoroso@gmail.com).
|
|
|
|
// All rights reserved.
|
|
|
|
|
2024-03-26 07:00:53 +01:00
|
|
|
// parser.go
|
|
|
|
package expr
|
|
|
|
|
|
|
|
import (
|
2024-04-02 04:36:03 +02:00
|
|
|
"errors"
|
2024-03-26 07:00:53 +01:00
|
|
|
"fmt"
|
|
|
|
)
|
|
|
|
|
|
|
|
//-------- parser
|
|
|
|
|
|
|
|
type parser struct {
|
2024-04-08 23:17:56 +02:00
|
|
|
ctx ExprContext
|
2024-03-26 07:00:53 +01:00
|
|
|
}
|
|
|
|
|
2024-04-08 23:17:56 +02:00
|
|
|
func NewParser(ctx ExprContext) (p *parser) {
|
2024-03-26 07:00:53 +01:00
|
|
|
p = &parser{
|
|
|
|
ctx: ctx,
|
|
|
|
}
|
|
|
|
return p
|
|
|
|
}
|
|
|
|
|
2024-04-04 12:54:26 +02:00
|
|
|
func (self *parser) parseFuncCall(scanner *scanner, allowVarRef bool, tk *Token) (tree *term, err error) {
|
2024-04-02 04:36:03 +02:00
|
|
|
// name, _ := tk.Value.(string)
|
|
|
|
// funcObj := self.ctx.GetFuncInfo(name)
|
|
|
|
// if funcObj == nil {
|
|
|
|
// err = fmt.Errorf("unknown function %s()", name)
|
|
|
|
// return
|
|
|
|
// }
|
|
|
|
// maxArgs := funcObj.MaxArgs()
|
|
|
|
// if maxArgs < 0 {
|
|
|
|
// maxArgs = funcObj.MinArgs() + 10
|
|
|
|
// }
|
|
|
|
// args := make([]*term, 0, maxArgs)
|
|
|
|
args := make([]*term, 0, 10)
|
2024-03-28 08:51:02 +01:00
|
|
|
lastSym := SymUnknown
|
|
|
|
for lastSym != SymClosedRound && lastSym != SymEos {
|
|
|
|
var subTree *ast
|
2024-04-04 12:54:26 +02:00
|
|
|
if subTree, err = self.parseItem(scanner, allowVarRef, SymComma, SymClosedRound); err == nil {
|
2024-03-28 08:51:02 +01:00
|
|
|
if subTree.root != nil {
|
|
|
|
args = append(args, subTree.root)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
lastSym = scanner.Previous().Sym
|
|
|
|
}
|
|
|
|
if err == nil {
|
|
|
|
// TODO Check arguments
|
2024-04-02 04:36:03 +02:00
|
|
|
if lastSym != SymClosedRound {
|
|
|
|
err = errors.New("unterminate arguments list")
|
|
|
|
} else {
|
|
|
|
tree = newFuncCallTerm(tk, args)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-04-26 04:45:42 +02:00
|
|
|
// func (self *parser) parseFuncDef(scanner *scanner) (tree *term, err error) {
|
|
|
|
// // Example: "add = func(x,y) {x+y}
|
|
|
|
// var body *ast
|
|
|
|
// args := make([]*term, 0)
|
|
|
|
// tk := scanner.Next()
|
|
|
|
// for tk.Sym != SymClosedRound && tk.Sym != SymEos {
|
|
|
|
// if tk.Sym == SymIdentifier {
|
|
|
|
// t := newTerm(tk, nil)
|
|
|
|
// args = append(args, t)
|
|
|
|
// } else {
|
|
|
|
// err = tk.Errorf("invalid param %q, variable identifier expected", tk.source)
|
|
|
|
// break
|
|
|
|
// }
|
|
|
|
// tk = scanner.Next()
|
|
|
|
// }
|
|
|
|
// if err == nil && tk.Sym != SymClosedRound {
|
|
|
|
// err = tk.Errorf("unterminate function params list")
|
|
|
|
// }
|
|
|
|
// if err == nil {
|
|
|
|
// tk = scanner.Next()
|
|
|
|
// if tk.Sym == SymOpenBrace {
|
|
|
|
// body, err = self.parseGeneral(scanner, true, true, SymClosedBrace)
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// if err == nil {
|
|
|
|
// // TODO Check arguments
|
|
|
|
// if scanner.Previous().Sym != SymClosedBrace {
|
|
|
|
// err = scanner.Previous().Errorf("not properly terminated function body")
|
|
|
|
// } else {
|
|
|
|
// tk = scanner.makeValueToken(SymExpression, "", body)
|
|
|
|
// tree = newFuncDefTerm(tk, args)
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// return
|
|
|
|
// }
|
|
|
|
|
2024-04-02 04:36:03 +02:00
|
|
|
func (self *parser) parseFuncDef(scanner *scanner) (tree *term, err error) {
|
|
|
|
// Example: "add = func(x,y) {x+y}
|
|
|
|
var body *ast
|
|
|
|
args := make([]*term, 0)
|
2024-04-26 04:45:42 +02:00
|
|
|
lastSym := SymUnknown
|
|
|
|
itemExpected := false
|
|
|
|
tk := scanner.Previous()
|
|
|
|
for lastSym != SymClosedRound && lastSym != SymEos {
|
|
|
|
var subTree *ast
|
|
|
|
if subTree, err = self.parseItem(scanner, true, SymComma, SymClosedRound); err == nil {
|
|
|
|
if subTree.root != nil {
|
|
|
|
if subTree.root.symbol() == SymIdentifier {
|
|
|
|
args = append(args, subTree.root)
|
|
|
|
} else {
|
|
|
|
err = tk.Errorf("exptected identifier, got %q", subTree.root)
|
|
|
|
}
|
|
|
|
} else if itemExpected {
|
|
|
|
prev := scanner.Previous()
|
|
|
|
err = prev.Errorf("expected function parameter, got %q", prev)
|
|
|
|
break
|
|
|
|
}
|
2024-04-02 04:36:03 +02:00
|
|
|
} else {
|
|
|
|
break
|
|
|
|
}
|
2024-04-26 04:45:42 +02:00
|
|
|
lastSym = scanner.Previous().Sym
|
|
|
|
itemExpected = lastSym == SymComma
|
2024-04-02 04:36:03 +02:00
|
|
|
}
|
2024-04-26 04:45:42 +02:00
|
|
|
|
|
|
|
if err == nil && lastSym != SymClosedRound {
|
|
|
|
err = tk.Errorf("unterminated function parameters list")
|
2024-04-02 04:36:03 +02:00
|
|
|
}
|
|
|
|
if err == nil {
|
|
|
|
tk = scanner.Next()
|
|
|
|
if tk.Sym == SymOpenBrace {
|
2024-04-04 12:54:26 +02:00
|
|
|
body, err = self.parseGeneral(scanner, true, true, SymClosedBrace)
|
2024-04-02 04:36:03 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if err == nil {
|
|
|
|
// TODO Check arguments
|
|
|
|
if scanner.Previous().Sym != SymClosedBrace {
|
2024-04-04 12:54:26 +02:00
|
|
|
err = scanner.Previous().Errorf("not properly terminated function body")
|
2024-04-02 04:36:03 +02:00
|
|
|
} else {
|
|
|
|
tk = scanner.makeValueToken(SymExpression, "", body)
|
|
|
|
tree = newFuncDefTerm(tk, args)
|
|
|
|
}
|
2024-03-28 08:51:02 +01:00
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-04-08 22:16:07 +02:00
|
|
|
func (self *parser) parseList(scanner *scanner, allowVarRef bool) (subtree *term, err error) {
|
2024-03-28 08:51:02 +01:00
|
|
|
args := make([]*term, 0)
|
|
|
|
lastSym := SymUnknown
|
2024-04-21 14:24:56 +02:00
|
|
|
itemExpected := false
|
2024-03-28 08:51:02 +01:00
|
|
|
for lastSym != SymClosedSquare && lastSym != SymEos {
|
|
|
|
var subTree *ast
|
2024-04-04 12:54:26 +02:00
|
|
|
if subTree, err = self.parseItem(scanner, allowVarRef, SymComma, SymClosedSquare); err == nil {
|
2024-03-28 08:51:02 +01:00
|
|
|
if subTree.root != nil {
|
|
|
|
args = append(args, subTree.root)
|
2024-04-21 14:24:56 +02:00
|
|
|
} else if itemExpected {
|
|
|
|
prev := scanner.Previous()
|
|
|
|
err = prev.Errorf("expected list item, got %q", prev)
|
|
|
|
break
|
2024-03-28 08:51:02 +01:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
lastSym = scanner.Previous().Sym
|
2024-04-21 14:24:56 +02:00
|
|
|
itemExpected = lastSym == SymComma
|
2024-03-28 08:51:02 +01:00
|
|
|
}
|
|
|
|
if err == nil {
|
|
|
|
// TODO Check arguments
|
2024-04-02 04:36:03 +02:00
|
|
|
if lastSym != SymClosedSquare {
|
|
|
|
err = scanner.Previous().Errorf("unterminate items list")
|
|
|
|
} else {
|
2024-04-08 22:16:07 +02:00
|
|
|
subtree = newListTerm(args)
|
2024-04-02 04:36:03 +02:00
|
|
|
}
|
2024-03-28 08:51:02 +01:00
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-04-26 04:45:42 +02:00
|
|
|
func (self *parser) parseIterDef(scanner *scanner, allowVarRef bool) (subtree *term, err error) {
|
|
|
|
var ds *term
|
|
|
|
tk := scanner.Previous()
|
|
|
|
args := make([]*term, 0)
|
|
|
|
lastSym := SymUnknown
|
|
|
|
dsExpected := true
|
|
|
|
itemExpected := false
|
|
|
|
for lastSym != SymClosedRound && lastSym != SymEos {
|
|
|
|
var subTree *ast
|
|
|
|
if subTree, err = self.parseItem(scanner, allowVarRef, SymComma, SymClosedRound); err == nil {
|
|
|
|
if subTree.root != nil {
|
|
|
|
if dsExpected {
|
|
|
|
if sym := subTree.root.symbol(); sym == SymDict || sym == SymIdentifier {
|
|
|
|
ds = subTree.root
|
|
|
|
} else {
|
|
|
|
err = subTree.root.Errorf("data-source dictionary expected, got %q", subTree.root.source())
|
|
|
|
}
|
|
|
|
dsExpected = false
|
|
|
|
} else {
|
|
|
|
args = append(args, subTree.root)
|
|
|
|
}
|
|
|
|
} else if itemExpected {
|
|
|
|
prev := scanner.Previous()
|
|
|
|
err = prev.Errorf("expected iterator argument, got %q", prev)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
lastSym = scanner.Previous().Sym
|
|
|
|
itemExpected = lastSym == SymComma
|
|
|
|
}
|
|
|
|
if err == nil {
|
|
|
|
// TODO Check arguments
|
|
|
|
if lastSym != SymClosedRound {
|
|
|
|
err = scanner.Previous().Errorf("unterminate iterator param list")
|
|
|
|
} else if ds != nil {
|
|
|
|
subtree = newIteratorTerm(tk, ds, args)
|
|
|
|
} else {
|
|
|
|
tk.Errorf("missing data-source param")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-04-21 14:24:56 +02:00
|
|
|
func (self *parser) parseDictKey(scanner *scanner, allowVarRef bool) (key any, err error) {
|
|
|
|
tk := scanner.Next()
|
|
|
|
if tk.Sym == SymError {
|
|
|
|
err = tk.Error()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if tk.Sym == SymClosedBrace || tk.Sym == SymEos {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if tk.Sym == SymInteger || tk.Sym == SymString {
|
|
|
|
tkSep := scanner.Next()
|
|
|
|
if tkSep.Sym != SymColon {
|
|
|
|
err = tkSep.Errorf("expected \":\", got %q", tkSep)
|
|
|
|
} else {
|
|
|
|
key = tk.Value
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
err = tk.Errorf("expected dictionary key or closed brace, got %q", tk)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *parser) parseDictionary(scanner *scanner, allowVarRef bool) (subtree *term, err error) {
|
|
|
|
args := make(map[any]*term, 0)
|
|
|
|
lastSym := SymUnknown
|
|
|
|
itemExpected := false
|
|
|
|
for lastSym != SymClosedBrace && lastSym != SymEos {
|
|
|
|
var subTree *ast
|
|
|
|
var key any
|
|
|
|
if key, err = self.parseDictKey(scanner, allowVarRef); err != nil {
|
|
|
|
break
|
2024-04-26 04:45:42 +02:00
|
|
|
} else if key == nil {
|
2024-04-21 14:24:56 +02:00
|
|
|
tk := scanner.Previous()
|
2024-04-26 04:45:42 +02:00
|
|
|
lastSym = tk.Sym
|
|
|
|
if itemExpected {
|
|
|
|
err = tk.Errorf("expected dictionary key, got %q", tk)
|
|
|
|
}
|
2024-04-21 14:24:56 +02:00
|
|
|
break
|
|
|
|
}
|
|
|
|
if subTree, err = self.parseItem(scanner, allowVarRef, SymComma, SymClosedBrace); err == nil {
|
|
|
|
if subTree.root != nil {
|
|
|
|
args[key] = subTree.root
|
|
|
|
} else if key != nil {
|
|
|
|
prev := scanner.Previous()
|
|
|
|
err = prev.Errorf("expected dictionary value, got %q", prev)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
lastSym = scanner.Previous().Sym
|
|
|
|
itemExpected = lastSym == SymComma
|
|
|
|
}
|
|
|
|
if err == nil {
|
|
|
|
// TODO Check arguments
|
|
|
|
if lastSym != SymClosedBrace {
|
2024-04-26 04:45:42 +02:00
|
|
|
err = scanner.Previous().Errorf("unterminated dictionary")
|
2024-04-21 14:24:56 +02:00
|
|
|
} else {
|
|
|
|
subtree = newDictTerm(args)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-04-08 22:16:07 +02:00
|
|
|
func (self *parser) parseSelectorCase(scanner *scanner, allowVarRef bool, defaultCase bool) (caseTerm *term, err error) {
|
|
|
|
var filterList *term
|
|
|
|
var caseExpr *ast
|
|
|
|
tk := scanner.Next()
|
|
|
|
startRow := tk.row
|
|
|
|
startCol := tk.col
|
|
|
|
if tk.Sym == SymOpenSquare {
|
|
|
|
if defaultCase {
|
|
|
|
err = tk.Errorf("case list in default clause")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if filterList, err = self.parseList(scanner, allowVarRef); err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
tk = scanner.Next()
|
|
|
|
startRow = tk.row
|
|
|
|
startCol = tk.col
|
|
|
|
} else if !defaultCase {
|
|
|
|
filterList = newListTerm(make([]*term, 0))
|
|
|
|
}
|
|
|
|
|
|
|
|
if tk.Sym == SymOpenBrace {
|
|
|
|
if caseExpr, err = self.parseGeneral(scanner, true, allowVarRef, SymClosedBrace); err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
err = tk.Errorf("selector-case expected, got %q", tk.source)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err == nil {
|
|
|
|
caseTerm = newSelectorCaseTerm(startRow, startCol, filterList, caseExpr)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-04-13 10:10:25 +02:00
|
|
|
func addSelectorCase(selectorTerm, caseTerm *term) {
|
|
|
|
if len(selectorTerm.children) < 2 {
|
|
|
|
caseListTerm := newListTermA(caseTerm)
|
|
|
|
selectorTerm.children = append(selectorTerm.children, caseListTerm)
|
|
|
|
} else {
|
|
|
|
caseListTerm := selectorTerm.children[1]
|
|
|
|
caseList, _ := caseListTerm.value().([]*term)
|
|
|
|
caseList = append(caseList, caseTerm)
|
|
|
|
caseListTerm.tk.Value = caseList
|
|
|
|
}
|
|
|
|
caseTerm.parent = selectorTerm
|
|
|
|
}
|
|
|
|
|
2024-04-08 22:16:07 +02:00
|
|
|
func (self *parser) parseSelector(scanner *scanner, tree *ast, allowVarRef bool) (selectorTerm *term, err error) {
|
|
|
|
var caseTerm *term
|
|
|
|
tk := scanner.makeToken(SymSelector, '?')
|
|
|
|
if selectorTerm, err = tree.addToken2(tk); err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if caseTerm, err = self.parseSelectorCase(scanner, allowVarRef, false); err == nil {
|
2024-04-13 10:10:25 +02:00
|
|
|
addSelectorCase(selectorTerm, caseTerm)
|
2024-04-08 22:16:07 +02:00
|
|
|
}
|
|
|
|
return
|
2024-04-02 04:36:03 +02:00
|
|
|
}
|
|
|
|
|
2024-04-04 12:54:26 +02:00
|
|
|
func (self *parser) parseItem(scanner *scanner, allowVarRef bool, termSymbols ...Symbol) (tree *ast, err error) {
|
|
|
|
return self.parseGeneral(scanner, false, allowVarRef, termSymbols...)
|
2024-04-02 04:36:03 +02:00
|
|
|
}
|
|
|
|
|
2024-04-08 22:16:07 +02:00
|
|
|
func (self *parser) Parse(scanner *scanner, termSymbols ...Symbol) (tree *ast, err error) {
|
|
|
|
return self.parseGeneral(scanner, true, false, termSymbols...)
|
|
|
|
}
|
|
|
|
|
2024-04-04 12:54:26 +02:00
|
|
|
func (self *parser) parseGeneral(scanner *scanner, allowForest bool, allowVarRef bool, termSymbols ...Symbol) (tree *ast, err error) {
|
2024-04-08 22:16:07 +02:00
|
|
|
var selectorTerm *term = nil
|
|
|
|
var currentTerm *term = nil
|
2024-04-17 14:15:50 +02:00
|
|
|
var tk *Token
|
2024-03-26 07:00:53 +01:00
|
|
|
tree = NewAst()
|
|
|
|
firstToken := true
|
2024-03-31 06:38:46 +02:00
|
|
|
lastSym := SymUnknown
|
2024-04-17 14:15:50 +02:00
|
|
|
for tk = scanner.Next(); err == nil && tk != nil && !tk.IsTerm(termSymbols); tk = scanner.Next() {
|
2024-03-26 07:00:53 +01:00
|
|
|
if tk.Sym == SymComment {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2024-03-31 06:10:27 +02:00
|
|
|
if tk.Sym == SymSemiColon {
|
2024-04-04 12:54:26 +02:00
|
|
|
if allowForest {
|
2024-04-02 04:36:03 +02:00
|
|
|
tree.ToForest()
|
2024-04-06 01:32:29 +02:00
|
|
|
firstToken = true
|
2024-04-02 04:36:03 +02:00
|
|
|
continue
|
|
|
|
} else {
|
|
|
|
err = tk.Errorf(`unexpected token %q, expected ",", "]", or ")"`, tk.source)
|
|
|
|
break
|
|
|
|
}
|
2024-03-31 06:10:27 +02:00
|
|
|
}
|
|
|
|
|
2024-03-26 07:00:53 +01:00
|
|
|
//fmt.Println("Token:", tk)
|
2024-04-02 04:36:03 +02:00
|
|
|
if firstToken {
|
2024-03-26 07:00:53 +01:00
|
|
|
if tk.Sym == SymMinus {
|
|
|
|
tk.Sym = SymChangeSign
|
2024-04-02 04:36:03 +02:00
|
|
|
} else if tk.Sym == SymPlus {
|
2024-03-26 07:00:53 +01:00
|
|
|
tk.Sym = SymUnchangeSign
|
|
|
|
}
|
2024-04-02 04:36:03 +02:00
|
|
|
firstToken = false
|
2024-03-26 07:00:53 +01:00
|
|
|
}
|
2024-04-02 04:36:03 +02:00
|
|
|
|
2024-03-28 08:51:02 +01:00
|
|
|
switch tk.Sym {
|
|
|
|
case SymOpenRound:
|
2024-03-26 07:00:53 +01:00
|
|
|
var subTree *ast
|
2024-04-04 12:54:26 +02:00
|
|
|
if subTree, err = self.parseGeneral(scanner, false, allowVarRef, SymClosedRound); err == nil {
|
2024-03-26 07:00:53 +01:00
|
|
|
subTree.root.priority = priValue
|
2024-04-08 22:16:07 +02:00
|
|
|
err = tree.addTerm(subTree.root)
|
|
|
|
currentTerm = subTree.root
|
2024-03-26 07:00:53 +01:00
|
|
|
}
|
2024-04-02 04:36:03 +02:00
|
|
|
case SymFuncCall:
|
|
|
|
var funcCallTerm *term
|
2024-04-04 12:54:26 +02:00
|
|
|
if funcCallTerm, err = self.parseFuncCall(scanner, allowVarRef, tk); err == nil {
|
2024-04-02 04:36:03 +02:00
|
|
|
err = tree.addTerm(funcCallTerm)
|
2024-04-08 22:16:07 +02:00
|
|
|
currentTerm = funcCallTerm
|
2024-03-26 07:00:53 +01:00
|
|
|
}
|
2024-03-28 08:51:02 +01:00
|
|
|
case SymOpenSquare:
|
|
|
|
var listTerm *term
|
2024-04-04 12:54:26 +02:00
|
|
|
if listTerm, err = self.parseList(scanner, allowVarRef); err == nil {
|
2024-03-28 08:51:02 +01:00
|
|
|
err = tree.addTerm(listTerm)
|
2024-04-08 22:16:07 +02:00
|
|
|
currentTerm = listTerm
|
2024-03-26 07:00:53 +01:00
|
|
|
}
|
2024-04-21 14:24:56 +02:00
|
|
|
case SymOpenBrace:
|
|
|
|
var mapTerm *term
|
|
|
|
if mapTerm, err = self.parseDictionary(scanner, allowVarRef); err == nil {
|
|
|
|
err = tree.addTerm(mapTerm)
|
|
|
|
currentTerm = mapTerm
|
|
|
|
}
|
2024-03-31 06:38:46 +02:00
|
|
|
case SymEqual:
|
2024-04-02 04:36:03 +02:00
|
|
|
if err = checkPrevSymbol(lastSym, SymIdentifier, tk); err == nil {
|
2024-04-08 22:16:07 +02:00
|
|
|
currentTerm, err = tree.addToken2(tk)
|
2024-04-02 04:36:03 +02:00
|
|
|
}
|
|
|
|
case SymFuncDef:
|
|
|
|
var funcDefTerm *term
|
|
|
|
if funcDefTerm, err = self.parseFuncDef(scanner); err == nil {
|
|
|
|
err = tree.addTerm(funcDefTerm)
|
2024-04-08 22:16:07 +02:00
|
|
|
currentTerm = funcDefTerm
|
2024-03-31 06:38:46 +02:00
|
|
|
}
|
2024-04-26 04:45:42 +02:00
|
|
|
case SymDollarRound:
|
|
|
|
var iterDefTerm *term
|
|
|
|
if iterDefTerm, err = self.parseIterDef(scanner, allowVarRef); err == nil {
|
|
|
|
err = tree.addTerm(iterDefTerm)
|
|
|
|
currentTerm = iterDefTerm
|
|
|
|
}
|
2024-04-04 12:54:26 +02:00
|
|
|
case SymIdentifier:
|
|
|
|
if tk.source[0] == '@' && !allowVarRef {
|
|
|
|
err = tk.Errorf("variable references are not allowed in top level expressions: %q", tk.source)
|
|
|
|
} else {
|
2024-04-08 22:16:07 +02:00
|
|
|
currentTerm, err = tree.addToken2(tk)
|
|
|
|
}
|
|
|
|
case SymQuestion:
|
2024-04-13 10:10:25 +02:00
|
|
|
if selectorTerm, err = self.parseSelector(scanner, tree, allowVarRef); err == nil {
|
2024-04-08 22:16:07 +02:00
|
|
|
currentTerm = selectorTerm
|
|
|
|
}
|
|
|
|
case SymColon, SymDoubleColon:
|
|
|
|
var caseTerm *term
|
|
|
|
if selectorTerm == nil {
|
|
|
|
err = tk.Errorf("selector-case outside of a selector context")
|
|
|
|
} else if caseTerm, err = self.parseSelectorCase(scanner, allowVarRef, tk.Sym == SymDoubleColon); err == nil {
|
2024-04-13 10:10:25 +02:00
|
|
|
addSelectorCase(selectorTerm, caseTerm)
|
2024-04-08 22:16:07 +02:00
|
|
|
currentTerm = caseTerm
|
2024-04-08 23:17:56 +02:00
|
|
|
if tk.Sym == SymDoubleColon {
|
|
|
|
selectorTerm = nil
|
|
|
|
}
|
2024-04-04 12:54:26 +02:00
|
|
|
}
|
2024-03-28 08:51:02 +01:00
|
|
|
default:
|
2024-04-08 22:16:07 +02:00
|
|
|
currentTerm, err = tree.addToken2(tk)
|
|
|
|
}
|
|
|
|
|
2024-04-13 04:37:15 +02:00
|
|
|
if currentTerm != nil && currentTerm.tk.Sym != SymSelector && currentTerm.parent != nil && currentTerm.parent.tk.Sym != SymSelector {
|
2024-04-08 22:16:07 +02:00
|
|
|
selectorTerm = nil
|
|
|
|
|
2024-03-26 07:00:53 +01:00
|
|
|
}
|
2024-03-31 06:38:46 +02:00
|
|
|
lastSym = tk.Sym
|
2024-03-26 07:00:53 +01:00
|
|
|
}
|
2024-04-17 14:15:50 +02:00
|
|
|
if err == nil {
|
|
|
|
err = tk.Error()
|
|
|
|
}
|
2024-03-26 07:00:53 +01:00
|
|
|
return
|
|
|
|
}
|
2024-04-02 04:36:03 +02:00
|
|
|
|
|
|
|
func checkPrevSymbol(lastSym, wantedSym Symbol, tk *Token) (err error) {
|
|
|
|
if lastSym != wantedSym {
|
|
|
|
err = fmt.Errorf(`assign operator (%q) must be preceded by a variable`, tk.source)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|