aboutsummaryrefslogtreecommitdiff
path: root/evaluator.go
blob: 486d45fb416e40508c5aaa78c4c53555ab08c752 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
package expression

import (
	"go/scanner"
	"go/token"
)

type Evaluator struct {
	operators map[token.Token]Operator
	functions map[string]func(stack *Stack) error
}

func New() *Evaluator {
	return &Evaluator{
		operators: DefaultOperators,
		functions: DefaultFunctions,
	}
}

func (e *Evaluator) Eval(expression string) (any, error) {
	s := scanner.Scanner{}
	fset := token.NewFileSet()
	file := fset.AddFile("", fset.Base(), len(expression))
	s.Init(file, []byte(expression), nil, scanner.ScanComments)
	tokens := make(chan Token)
	go func() {
		for {
			pos, tok, lit := s.Scan()
			if tok == token.SEMICOLON {
				continue
			}
			if tok == token.EOF {
				break
			}
			tokens <- Token{
				Token:   tok,
				Literal: lit,
				Pos:     int(pos),
			}
		}
		close(tokens)
	}()
	rpnTokens := e.ToPRN(tokens)
	return e.execute(rpnTokens)
}

type Operator struct {
	fn          func(stack *Stack) error
	priority    int
	isLeftAssoc bool
}