glox/interpreter.go

278 lines
5.2 KiB
Go
Raw Normal View History

2024-10-04 15:24:01 +03:00
package main
import (
"fmt"
2024-10-05 18:58:49 +03:00
"log"
2024-10-04 15:24:01 +03:00
"reflect"
2024-10-06 16:30:57 +03:00
"slices"
2024-10-04 15:24:01 +03:00
)
2024-10-05 18:58:49 +03:00
type Interpreter struct {
2024-10-05 23:27:00 +03:00
env *Environment
2024-10-05 18:58:49 +03:00
errors []error
}
2024-10-04 15:24:01 +03:00
type RuntimeError struct {
token Token
msg string
}
2024-10-05 18:58:49 +03:00
func (re *RuntimeError) Error() string {
return fmt.Sprintf("RuntimeError [%d][%s] Error: %s", re.token.line, re.token.typ, re.msg)
2024-10-04 15:24:01 +03:00
}
func newInterpreter() *Interpreter {
2024-10-05 23:27:00 +03:00
return &Interpreter{env: newEnvironment(nil)}
2024-10-04 15:24:01 +03:00
}
2024-10-05 18:58:49 +03:00
func (i *Interpreter) interpret(stmts []Stmt) []error {
2024-10-04 15:24:01 +03:00
defer i.recover()
2024-10-05 18:58:49 +03:00
i.errors = []error{}
for _, stmt := range stmts {
stmt.accept(i)
}
return i.errors
2024-10-04 15:24:01 +03:00
}
func (i *Interpreter) recover() {
if err := recover(); err != nil {
2024-10-05 18:58:49 +03:00
_, ok := err.(*RuntimeError)
2024-10-04 15:24:01 +03:00
if !ok {
panic(err)
}
}
}
2024-10-05 18:58:49 +03:00
func (i *Interpreter) evaluate(e Expr) any {
return e.accept(i)
}
2024-10-04 15:24:01 +03:00
func (i *Interpreter) visitBinary(b *Binary) any {
left := i.evaluate(b.left)
right := i.evaluate(b.right)
switch b.op.typ {
case MINUS:
2024-10-05 18:58:49 +03:00
i.checkIfFloats(b.op, left, right)
2024-10-04 15:24:01 +03:00
return left.(float64) - right.(float64)
case SLASH:
2024-10-05 18:58:49 +03:00
i.checkIfFloats(b.op, left, right)
2024-10-04 15:24:01 +03:00
return left.(float64) / right.(float64)
case STAR:
2024-10-05 18:58:49 +03:00
i.checkIfFloats(b.op, left, right)
2024-10-04 15:24:01 +03:00
return left.(float64) * right.(float64)
case GREATER:
2024-10-05 18:58:49 +03:00
i.checkIfFloats(b.op, left, right)
2024-10-04 15:24:01 +03:00
return left.(float64) > right.(float64)
case LESS:
2024-10-05 18:58:49 +03:00
i.checkIfFloats(b.op, left, right)
2024-10-04 15:24:01 +03:00
return left.(float64) < right.(float64)
case GREATER_EQUAL:
2024-10-05 18:58:49 +03:00
i.checkIfFloats(b.op, left, right)
2024-10-04 15:24:01 +03:00
return left.(float64) >= right.(float64)
case LESS_EQUAL:
2024-10-05 18:58:49 +03:00
i.checkIfFloats(b.op, left, right)
2024-10-04 15:24:01 +03:00
return left.(float64) <= right.(float64)
case BANG_EQUAL:
return !reflect.DeepEqual(left, right)
case EQUAL_EQUAL:
return reflect.DeepEqual(left, right)
case PLUS:
if isFloats(left, right) {
return left.(float64) + right.(float64)
}
if isStrings(left, right) {
return left.(string) + right.(string)
}
}
2024-10-05 18:58:49 +03:00
i.panic(&RuntimeError{b.op, fmt.Sprintf("Operands must be numbers or strings: %v %s %v", left, b.op.lexeme, right)})
2024-10-04 15:24:01 +03:00
return nil
}
func (i *Interpreter) visitLiteral(l *Literal) any {
return l.value
}
func (i *Interpreter) visitGrouping(g *Grouping) any {
return i.evaluate(g.expression)
}
func (i *Interpreter) visitUnary(u *Unary) any {
val := i.evaluate(u.right)
switch u.op.typ {
case MINUS:
2024-10-05 18:58:49 +03:00
i.checkIfFloat(u.op, val)
2024-10-04 15:24:01 +03:00
return -val.(float64)
case BANG:
return !isTruthy(val)
}
return nil
}
2024-10-05 18:58:49 +03:00
func (i *Interpreter) visitVariable(v *Variable) any {
2024-10-06 16:30:57 +03:00
if !i.env.exists(v.name.lexeme) {
i.panic(&RuntimeError{v.name, fmt.Sprintf("Can't assign: undefined variable '%s'.", v.name.lexeme)})
2024-10-05 23:27:00 +03:00
}
2024-10-06 16:30:57 +03:00
val := i.env.get(v.name.lexeme)
2024-10-05 23:27:00 +03:00
return val
2024-10-05 18:58:49 +03:00
}
func (i *Interpreter) visitAssignment(a *Assign) any {
2024-10-05 23:27:00 +03:00
if !i.env.exists(a.variable.lexeme) {
i.panic(&RuntimeError{a.variable, fmt.Sprintf("Can't assign: undefined variable '%s'.", a.variable.lexeme)})
2024-10-05 18:58:49 +03:00
}
2024-10-05 23:27:00 +03:00
val := i.evaluate(a.value)
2024-10-05 18:58:49 +03:00
2024-10-05 23:27:00 +03:00
i.env.set(a.variable.lexeme, val)
return val
2024-10-05 18:58:49 +03:00
}
2024-10-06 17:15:50 +03:00
func (i *Interpreter) visitLogical(lo *Logical) any {
left := i.evaluate(lo.left)
shortOr := lo.operator.typ == OR && isTruthy(left)
shortAnd := lo.operator.typ == AND && !isTruthy(left)
if shortOr || shortAnd {
return left
}
2024-10-06 16:30:57 +03:00
2024-10-06 17:15:50 +03:00
return i.evaluate(lo.right)
2024-10-06 16:30:57 +03:00
}
2024-10-05 18:58:49 +03:00
func (i *Interpreter) visitPrintStmt(p *PrintStmt) {
fmt.Printf("%v\n", i.evaluate(p.val))
}
func (i *Interpreter) visitExprStmt(se *ExprStmt) {
i.evaluate(se.expr)
}
func (i *Interpreter) visitVarStmt(v *VarStmt) {
var val any = nil
if v.initializer != nil {
val = i.evaluate(v.initializer)
}
2024-10-05 23:27:00 +03:00
i.env.set(v.name.lexeme, val)
}
func (i *Interpreter) visitBlockStmt(b *BlockStmt) {
parentEnv := i.env
i.env = newEnvironment(parentEnv)
for _, stmt := range b.stmts {
stmt.accept(i)
}
i.env = parentEnv
2024-10-05 18:58:49 +03:00
}
2024-10-06 16:30:57 +03:00
func (i *Interpreter) visitIfStmt(iff *IfStmt) {
if isTruthy(i.evaluate(iff.expr)) {
iff.then.accept(i)
} else if iff.or != nil {
iff.or.accept(i)
}
}
func (i *Interpreter) visitEnvStmt(e *EnvStmt) {
walker := i.env
flatten := []*Environment{}
for walker != nil {
flatten = slices.Insert(flatten, 0, walker)
walker = walker.parent
}
for ident, e := range flatten {
fmt.Printf("%*s", ident, "")
fmt.Printf("%+v\n", *e)
}
2024-10-07 21:06:23 +03:00
}
2024-10-06 16:30:57 +03:00
2024-10-07 21:06:23 +03:00
func (i *Interpreter) visitWhileStmt(w *WhileStmt) {
for isTruthy(i.evaluate(w.cond)) {
w.body.accept(i)
}
2024-10-06 16:30:57 +03:00
}
2024-10-05 18:58:49 +03:00
func (i *Interpreter) panic(re *RuntimeError) {
i.errors = append(i.errors, re)
log.Println(re)
panic(re)
}
func (i *Interpreter) checkIfFloat(op Token, val any) {
2024-10-04 15:24:01 +03:00
if _, ok := val.(float64); ok {
return
}
2024-10-05 18:58:49 +03:00
i.panic(&RuntimeError{op, "value must be a number."})
2024-10-04 15:24:01 +03:00
}
2024-10-05 18:58:49 +03:00
func (i *Interpreter) checkIfFloats(op Token, a any, b any) {
2024-10-04 15:24:01 +03:00
if isFloats(a, b) {
return
}
2024-10-05 18:58:49 +03:00
i.panic(&RuntimeError{op, fmt.Sprintf("Operands must be numbers: %v %s %v", a, op.lexeme, b)})
2024-10-04 15:24:01 +03:00
}
func isFloats(a any, b any) bool {
2024-10-06 16:30:57 +03:00
if a == nil || b == nil {
return false
}
2024-10-04 15:24:01 +03:00
ltype := reflect.TypeOf(a)
rtype := reflect.TypeOf(b)
return ltype.Kind() == rtype.Kind() && ltype.Kind() == reflect.Float64
}
func isStrings(a any, b any) bool {
2024-10-06 16:30:57 +03:00
if a == nil || b == nil {
return false
}
2024-10-04 15:24:01 +03:00
ltype := reflect.TypeOf(a)
rtype := reflect.TypeOf(b)
return ltype.Kind() == rtype.Kind() && ltype.Kind() == reflect.String
}
func isTruthy(val any) bool {
if val == nil {
return false
}
if boolean, ok := val.(bool); ok {
return boolean
}
return true
}