evaluator.go 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277
  1. package evaluator
  2. import (
  3. "fmt"
  4. "github/runnignwater/monkey/ast"
  5. "github/runnignwater/monkey/object"
  6. )
  7. var (
  8. NULL = &object.Null{}
  9. TRUE = &object.Boolean{Value: true}
  10. FALSE = &object.Boolean{Value: false}
  11. )
  12. func Eval(node ast.Node, env *object.Environment) object.Object {
  13. switch node := node.(type) {
  14. // Statements
  15. case *ast.Program:
  16. return evalProgram(node, env)
  17. case *ast.ExpressionStatement:
  18. return Eval(node.Expression, env)
  19. // Expression
  20. case *ast.IntegerLiteral:
  21. return &object.Integer{Value: node.Value}
  22. case *ast.Boolean:
  23. return nativeBooleanObject(node.Value)
  24. case *ast.PrefixExpression:
  25. right := Eval(node.Right, env)
  26. if isError(right) {
  27. return right
  28. }
  29. return evalPrefixExpression(node.Operator, right)
  30. case *ast.InfixExpression:
  31. left := Eval(node.Left, env)
  32. if isError(left) {
  33. return left
  34. }
  35. right := Eval(node.Right, env)
  36. if isError(right) {
  37. return right
  38. }
  39. return evalInfixExpression(node.Operator, left, right)
  40. case *ast.BlockStatement:
  41. return evalBlockStatements(node, env)
  42. case *ast.IfExpression:
  43. return evalIfExpression(node, env)
  44. case *ast.ReturnStatement:
  45. val := Eval(node.ReturnValue, env)
  46. if isError(val) {
  47. return val
  48. }
  49. return &object.ReturnValue{Value: val}
  50. case *ast.Identifier:
  51. return evalIdentifier(node, env)
  52. case *ast.LetStatement:
  53. val := Eval(node.Value, env)
  54. if isError(val) {
  55. return val
  56. }
  57. env.Set(node.Name.Value, val)
  58. case *ast.FunctionLiteral:
  59. params := node.Parameters
  60. body := node.Body
  61. return &object.Function{Parameters: params, Body: body, Env: env}
  62. case *ast.CallExpression:
  63. function := Eval(node.Function, env)
  64. if isError(function) {
  65. return function
  66. }
  67. args := evalExpressions(node.Arguments, env)
  68. if len(args) == 1 && isError(args[0]) {
  69. return args[0]
  70. }
  71. return applyFunction(function, args)
  72. }
  73. return nil
  74. }
  75. func applyFunction(fn object.Object, args []object.Object) object.Object {
  76. function, ok := fn.(*object.Function)
  77. if !ok {
  78. return newError("not a function: %s", fn.Type())
  79. }
  80. extendEnv := extendFunctionEnv(function, args)
  81. evaluated := Eval(function.Body, extendEnv)
  82. return unwrapReturnValue(evaluated)
  83. }
  84. func extendFunctionEnv(fn *object.Function, args []object.Object) *object.Environment {
  85. env := object.NewEnclosedEnvironment(fn.Env)
  86. for paramIdx, param := range fn.Parameters {
  87. env.Set(param.Value, args[paramIdx])
  88. }
  89. return env
  90. }
  91. func unwrapReturnValue(obj object.Object) object.Object {
  92. if returnValue, ok := obj.(*object.ReturnValue); ok {
  93. return returnValue.Value
  94. }
  95. return obj
  96. }
  97. func newError(format string, a ...interface{}) *object.Error {
  98. return &object.Error{Msg: fmt.Sprintf(format, a...)}
  99. }
  100. func evalProgram(node *ast.Program, env *object.Environment) object.Object {
  101. var result object.Object
  102. for _, stmt := range node.Statements {
  103. result = Eval(stmt, env)
  104. // skip return 后面的语句
  105. switch result := result.(type) {
  106. case *object.ReturnValue:
  107. return result.Value
  108. case *object.Error:
  109. return result
  110. }
  111. }
  112. return result
  113. }
  114. func nativeBooleanObject(input bool) *object.Boolean {
  115. if input {
  116. return TRUE
  117. } else {
  118. return FALSE
  119. }
  120. }
  121. func evalPrefixExpression(operator string, right object.Object) object.Object {
  122. switch operator {
  123. case "!":
  124. return evalBangOperatorExpression(right)
  125. case "-":
  126. return evalMinusPreOperatorExpression(right)
  127. default:
  128. return newError("unknown operator: %s%s", operator, right.Type())
  129. }
  130. }
  131. func evalInfixExpression(operator string, left object.Object, right object.Object) object.Object {
  132. switch {
  133. case left.Type() == object.IntegerObj && right.Type() == object.IntegerObj:
  134. return evalIntegerInfixExpression(operator, left, right)
  135. case operator == "==":
  136. return nativeBooleanObject(left == right)
  137. case operator == "!=":
  138. return nativeBooleanObject(left != right)
  139. case left.Type() != right.Type():
  140. return newError("type mismatch: %s %s %s", left.Type(), operator, right.Type())
  141. default:
  142. return newError("unknown operator: %s %s %s", left.Type(), operator, right.Type())
  143. }
  144. }
  145. func evalIfExpression(node *ast.IfExpression, env *object.Environment) object.Object {
  146. condition := Eval(node.Condition, env)
  147. if isError(condition) {
  148. return condition
  149. }
  150. if isTruthy(condition) {
  151. return Eval(node.Consequence, env)
  152. } else if node.Alternative != nil {
  153. return Eval(node.Alternative, env)
  154. } else {
  155. return NULL
  156. }
  157. }
  158. func evalBlockStatements(block *ast.BlockStatement, env *object.Environment) object.Object {
  159. var result object.Object
  160. for _, stmt := range block.Statements {
  161. result = Eval(stmt, env)
  162. if result != nil {
  163. rt := result.Type()
  164. if rt == object.ReturnValueObj || rt == object.ErrorObj {
  165. return result
  166. }
  167. }
  168. }
  169. return result
  170. }
  171. func evalIntegerInfixExpression(operator string,
  172. left object.Object, right object.Object) object.Object {
  173. leftVal := left.(*object.Integer).Value
  174. rightVal := right.(*object.Integer).Value
  175. switch operator {
  176. case "+":
  177. return &object.Integer{Value: leftVal + rightVal}
  178. case "-":
  179. return &object.Integer{Value: leftVal - rightVal}
  180. case "*":
  181. return &object.Integer{Value: leftVal * rightVal}
  182. case "/":
  183. return &object.Integer{Value: leftVal / rightVal}
  184. case "<":
  185. return nativeBooleanObject(leftVal < rightVal)
  186. case ">":
  187. return nativeBooleanObject(leftVal > rightVal)
  188. case "==":
  189. return nativeBooleanObject(leftVal == rightVal)
  190. case "!=":
  191. return nativeBooleanObject(leftVal != rightVal)
  192. default:
  193. return newError("unknown operator: %s %s %s", left.Type(), operator, right.Type())
  194. }
  195. }
  196. func evalMinusPreOperatorExpression(right object.Object) object.Object {
  197. if right.Type() != object.IntegerObj {
  198. return newError("unknown operator: -%s", right.Type())
  199. }
  200. value := right.(*object.Integer).Value
  201. return &object.Integer{Value: -value}
  202. }
  203. func evalBangOperatorExpression(right object.Object) object.Object {
  204. switch right {
  205. case TRUE:
  206. return FALSE
  207. case FALSE:
  208. return TRUE
  209. default:
  210. return FALSE
  211. }
  212. }
  213. func evalIdentifier(node *ast.Identifier, env *object.Environment) object.Object {
  214. val, ok := env.Get(node.Value)
  215. if !ok {
  216. return newError("identifier not found: " + node.Value)
  217. }
  218. return val
  219. }
  220. func evalExpressions(exps []ast.Expression, env *object.Environment) []object.Object {
  221. var result []object.Object
  222. for _, e := range exps {
  223. evaluated := Eval(e, env)
  224. if isError(evaluated) {
  225. return []object.Object{evaluated}
  226. }
  227. result = append(result, evaluated)
  228. }
  229. return result
  230. }
  231. func isTruthy(obj object.Object) bool {
  232. switch obj {
  233. case NULL:
  234. return false
  235. case TRUE:
  236. return true
  237. case FALSE:
  238. return false
  239. default:
  240. return true
  241. }
  242. }
  243. func isError(obj object.Object) bool {
  244. if obj != nil {
  245. return obj.Type() == object.ErrorObj
  246. }
  247. return false
  248. }