Environment and identifiers
This commit is contained in:
@@ -19,29 +19,39 @@ func isError(obj object.Object) bool {
|
||||
return false
|
||||
}
|
||||
|
||||
func Eval(node ast.Node) object.Object {
|
||||
func Eval(node ast.Node, env *object.Environment) object.Object {
|
||||
switch node := node.(type) {
|
||||
|
||||
// Statements
|
||||
case *ast.Program:
|
||||
return evalProgram(node)
|
||||
return evalProgram(node, env)
|
||||
|
||||
case *ast.ExpressionStatement:
|
||||
return Eval(node.Expression)
|
||||
return Eval(node.Expression, env)
|
||||
|
||||
case *ast.BlockStatement:
|
||||
return evalBlockStatements(node)
|
||||
return evalBlockStatements(node, env)
|
||||
|
||||
case *ast.IfExpression:
|
||||
return evalIfExpression(node)
|
||||
return evalIfExpression(node, env)
|
||||
|
||||
case *ast.ReturnStatement:
|
||||
val := Eval(node.ReturnValue)
|
||||
val := Eval(node.ReturnValue, env)
|
||||
if isError(val) {
|
||||
return val
|
||||
}
|
||||
return &object.ReturnValue{Value: val}
|
||||
|
||||
case *ast.LetStatement:
|
||||
val := Eval(node.Value, env)
|
||||
if isError(val) {
|
||||
return val
|
||||
}
|
||||
env.Set(node.Name.Value, val)
|
||||
|
||||
case *ast.Identifier:
|
||||
return evalIdentifier(node, env)
|
||||
|
||||
// Expressions
|
||||
case *ast.IntegerLiteral:
|
||||
return &object.Integer{Value: node.Value}
|
||||
@@ -50,18 +60,18 @@ func Eval(node ast.Node) object.Object {
|
||||
return nativeBoolToBooleanObject(node.Value)
|
||||
|
||||
case *ast.PrefixExpression:
|
||||
right := Eval(node.Right)
|
||||
right := Eval(node.Right, env)
|
||||
if isError(right) {
|
||||
return right
|
||||
}
|
||||
return evalPrefixExpression(node.Operator, right)
|
||||
|
||||
case *ast.InfixExpression:
|
||||
left := Eval(node.Left)
|
||||
left := Eval(node.Left, env)
|
||||
if isError(left) {
|
||||
return left
|
||||
}
|
||||
right := Eval(node.Right)
|
||||
right := Eval(node.Right, env)
|
||||
if isError(right) {
|
||||
return right
|
||||
}
|
||||
@@ -72,11 +82,11 @@ func Eval(node ast.Node) object.Object {
|
||||
return nil
|
||||
}
|
||||
|
||||
func evalProgram(program *ast.Program) object.Object {
|
||||
func evalProgram(program *ast.Program, env *object.Environment) object.Object {
|
||||
var result object.Object
|
||||
|
||||
for _, statement := range program.Statements {
|
||||
result = Eval(statement)
|
||||
result = Eval(statement, env)
|
||||
|
||||
switch result := result.(type) {
|
||||
case *object.ReturnValue:
|
||||
@@ -90,11 +100,11 @@ func evalProgram(program *ast.Program) object.Object {
|
||||
return result
|
||||
}
|
||||
|
||||
func evalBlockStatements(block *ast.BlockStatement) object.Object {
|
||||
func evalBlockStatements(block *ast.BlockStatement, env *object.Environment) object.Object {
|
||||
var result object.Object
|
||||
|
||||
for _, statement := range block.Statements {
|
||||
result = Eval(statement)
|
||||
result = Eval(statement, env)
|
||||
|
||||
if result != nil {
|
||||
rt := result.Type()
|
||||
@@ -188,16 +198,16 @@ func evalIntegerInfixExpression(operator string, left, right object.Object) obje
|
||||
}
|
||||
}
|
||||
|
||||
func evalIfExpression(ie *ast.IfExpression) object.Object {
|
||||
condition := Eval(ie.Condition)
|
||||
func evalIfExpression(ie *ast.IfExpression, env *object.Environment) object.Object {
|
||||
condition := Eval(ie.Condition, env)
|
||||
if isError(condition) {
|
||||
return condition
|
||||
}
|
||||
|
||||
if isTruthy(condition) {
|
||||
return Eval(ie.Consequence)
|
||||
return Eval(ie.Consequence, env)
|
||||
} else if ie.Alternative != nil {
|
||||
return Eval(ie.Alternative)
|
||||
return Eval(ie.Alternative, env)
|
||||
} else {
|
||||
return NULL
|
||||
}
|
||||
@@ -219,3 +229,11 @@ func isTruthy(obj object.Object) bool {
|
||||
func newError(format string, a ...interface{}) *object.Error {
|
||||
return &object.Error{Message: fmt.Sprintf(format, a...)}
|
||||
}
|
||||
|
||||
func evalIdentifier(node *ast.Identifier, env *object.Environment) object.Object {
|
||||
val, ok := env.Get(node.Value)
|
||||
if !ok {
|
||||
return newError("identifier not found: " + node.Value)
|
||||
}
|
||||
return val
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user