164 lines
3.4 KiB
Go
164 lines
3.4 KiB
Go
package evaluator
|
|
|
|
import (
|
|
"monkey/ast"
|
|
"monkey/object"
|
|
)
|
|
|
|
var (
|
|
NULL = &object.Null{}
|
|
TRUE = &object.Boolean{Value: true}
|
|
FALSE = &object.Boolean{Value: false}
|
|
)
|
|
|
|
func Eval(node ast.Node) object.Object {
|
|
switch node := node.(type) {
|
|
|
|
// Statements
|
|
case *ast.Program:
|
|
return evalStatements(node.Statements)
|
|
|
|
case *ast.ExpressionStatement:
|
|
return Eval(node.Expression)
|
|
|
|
case *ast.BlockStatement:
|
|
return evalStatements(node.Statements)
|
|
|
|
case *ast.IfExpression:
|
|
return evalIfExpression(node)
|
|
|
|
// Expressions
|
|
case *ast.IntegerLiteral:
|
|
return &object.Integer{Value: node.Value}
|
|
|
|
case *ast.Boolean:
|
|
return nativeBoolToBooleanObject(node.Value)
|
|
|
|
case *ast.PrefixExpression:
|
|
right := Eval(node.Right)
|
|
return evalPrefixExpression(node.Operator, right)
|
|
|
|
case *ast.InfixExpression:
|
|
left := Eval(node.Left)
|
|
right := Eval(node.Right)
|
|
return evalInfixExpression(node.Operator, left, right)
|
|
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func evalStatements(stmts []ast.Statement) object.Object {
|
|
var result object.Object
|
|
|
|
for _, statement := range stmts {
|
|
result = Eval(statement)
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
func nativeBoolToBooleanObject(input bool) object.Object {
|
|
if input {
|
|
return TRUE
|
|
}
|
|
return FALSE
|
|
}
|
|
|
|
func evalPrefixExpression(operator string, right object.Object) object.Object {
|
|
switch operator {
|
|
case "!":
|
|
return evalBangOperatorExpression(right)
|
|
case "-":
|
|
return evalMinusPrefixOperatorExpression(right)
|
|
default:
|
|
return NULL
|
|
}
|
|
}
|
|
|
|
func evalBangOperatorExpression(right object.Object) object.Object {
|
|
switch right {
|
|
case TRUE:
|
|
return FALSE
|
|
case FALSE:
|
|
return TRUE
|
|
case NULL:
|
|
return TRUE
|
|
default:
|
|
return FALSE
|
|
}
|
|
}
|
|
|
|
func evalMinusPrefixOperatorExpression(right object.Object) object.Object {
|
|
if right.Type() != object.INTEGER_OBJ {
|
|
return NULL
|
|
}
|
|
|
|
value := right.(*object.Integer).Value
|
|
return &object.Integer{Value: -value}
|
|
}
|
|
|
|
func evalInfixExpression(operator string, left, right object.Object) object.Object {
|
|
switch {
|
|
case left.Type() == object.INTEGER_OBJ && right.Type() == object.INTEGER_OBJ:
|
|
return evalIntegerInfixExpression(operator, left, right)
|
|
case operator == "==":
|
|
return nativeBoolToBooleanObject(left == right)
|
|
case operator == "!=":
|
|
return nativeBoolToBooleanObject(left != right)
|
|
default:
|
|
return NULL
|
|
}
|
|
}
|
|
|
|
func evalIntegerInfixExpression(operator string, left, right object.Object) object.Object {
|
|
leftVal := left.(*object.Integer).Value
|
|
rightVal := right.(*object.Integer).Value
|
|
|
|
switch operator {
|
|
case "+":
|
|
return &object.Integer{Value: leftVal + rightVal}
|
|
case "-":
|
|
return &object.Integer{Value: leftVal - rightVal}
|
|
case "*":
|
|
return &object.Integer{Value: leftVal * rightVal}
|
|
case "/":
|
|
return &object.Integer{Value: leftVal / rightVal}
|
|
case "<":
|
|
return nativeBoolToBooleanObject(leftVal < rightVal)
|
|
case ">":
|
|
return nativeBoolToBooleanObject(leftVal > rightVal)
|
|
case "==":
|
|
return nativeBoolToBooleanObject(leftVal == rightVal)
|
|
case "!=":
|
|
return nativeBoolToBooleanObject(leftVal != rightVal)
|
|
default:
|
|
return NULL
|
|
}
|
|
}
|
|
|
|
func evalIfExpression(ie *ast.IfExpression) object.Object {
|
|
condition := Eval(ie.Condition)
|
|
|
|
if isTruthy(condition) {
|
|
return Eval(ie.Consequence)
|
|
} else if ie.Alternative != nil {
|
|
return Eval(ie.Alternative)
|
|
} else {
|
|
return NULL
|
|
}
|
|
}
|
|
|
|
func isTruthy(obj object.Object) bool {
|
|
switch obj {
|
|
case NULL:
|
|
return false
|
|
case TRUE:
|
|
return true
|
|
case FALSE:
|
|
return false
|
|
default:
|
|
return true
|
|
}
|
|
}
|