evaluator.go 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334
  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. case *ast.StringLiteral:
  73. return &object.String{Value: node.Value}
  74. case *ast.ArrayLiteral:
  75. elements := evalExpressions(node.Element, env)
  76. if len(elements) == 1 && isError(elements[0]) {
  77. // return object.Error
  78. return elements[0]
  79. }
  80. return &object.Array{Elements: elements}
  81. case *ast.IndexExpression:
  82. left := Eval(node.Left, env)
  83. if isError(left) {
  84. return left
  85. }
  86. index := Eval(node.Index, env)
  87. if isError(index) {
  88. return index
  89. }
  90. return evalIndexExpression(left, index)
  91. }
  92. return nil
  93. }
  94. func applyFunction(fn object.Object, args []object.Object) object.Object {
  95. switch fn := fn.(type) {
  96. case *object.Function:
  97. extendEnv := extendFunctionEnv(fn, args)
  98. evaluated := Eval(fn.Body, extendEnv)
  99. return unwrapReturnValue(evaluated)
  100. case *object.Builtin:
  101. return fn.Fn(args...)
  102. default:
  103. return newError("not a function: %s", fn.Type())
  104. }
  105. }
  106. func extendFunctionEnv(fn *object.Function, args []object.Object) *object.Environment {
  107. env := object.NewEnclosedEnvironment(fn.Env)
  108. for paramIdx, param := range fn.Parameters {
  109. env.Set(param.Value, args[paramIdx])
  110. }
  111. return env
  112. }
  113. func unwrapReturnValue(obj object.Object) object.Object {
  114. if returnValue, ok := obj.(*object.ReturnValue); ok {
  115. return returnValue.Value
  116. }
  117. return obj
  118. }
  119. func newError(format string, a ...interface{}) *object.Error {
  120. return &object.Error{Msg: fmt.Sprintf(format, a...)}
  121. }
  122. func evalProgram(node *ast.Program, env *object.Environment) object.Object {
  123. var result object.Object
  124. for _, stmt := range node.Statements {
  125. result = Eval(stmt, env)
  126. // skip return 后面的语句
  127. switch result := result.(type) {
  128. case *object.ReturnValue:
  129. return result.Value
  130. case *object.Error:
  131. return result
  132. }
  133. }
  134. return result
  135. }
  136. func nativeBooleanObject(input bool) *object.Boolean {
  137. if input {
  138. return TRUE
  139. } else {
  140. return FALSE
  141. }
  142. }
  143. func evalPrefixExpression(operator string, right object.Object) object.Object {
  144. switch operator {
  145. case "!":
  146. return evalBangOperatorExpression(right)
  147. case "-":
  148. return evalMinusPreOperatorExpression(right)
  149. default:
  150. return newError("unknown operator: %s%s", operator, right.Type())
  151. }
  152. }
  153. func evalInfixExpression(operator string, left object.Object, right object.Object) object.Object {
  154. switch {
  155. case left.Type() == object.IntegerObj && right.Type() == object.IntegerObj:
  156. return evalIntegerInfixExpression(operator, left, right)
  157. case left.Type() == object.StringObj && right.Type() == object.StringObj:
  158. return evalStringInfixExpression(operator, left, right)
  159. case operator == "==":
  160. return nativeBooleanObject(left == right)
  161. case operator == "!=":
  162. return nativeBooleanObject(left != right)
  163. case left.Type() != right.Type():
  164. return newError("type mismatch: %s %s %s", left.Type(), operator, right.Type())
  165. default:
  166. return newError("unknown operator: %s %s %s", left.Type(), operator, right.Type())
  167. }
  168. }
  169. func evalIfExpression(node *ast.IfExpression, env *object.Environment) object.Object {
  170. condition := Eval(node.Condition, env)
  171. if isError(condition) {
  172. return condition
  173. }
  174. if isTruthy(condition) {
  175. return Eval(node.Consequence, env)
  176. } else if node.Alternative != nil {
  177. return Eval(node.Alternative, env)
  178. } else {
  179. return NULL
  180. }
  181. }
  182. func evalBlockStatements(block *ast.BlockStatement, env *object.Environment) object.Object {
  183. var result object.Object
  184. for _, stmt := range block.Statements {
  185. result = Eval(stmt, env)
  186. if result != nil {
  187. rt := result.Type()
  188. if rt == object.ReturnValueObj || rt == object.ErrorObj {
  189. return result
  190. }
  191. }
  192. }
  193. return result
  194. }
  195. func evalIntegerInfixExpression(operator string,
  196. left object.Object, right object.Object) object.Object {
  197. leftVal := left.(*object.Integer).Value
  198. rightVal := right.(*object.Integer).Value
  199. switch operator {
  200. case "+":
  201. return &object.Integer{Value: leftVal + rightVal}
  202. case "-":
  203. return &object.Integer{Value: leftVal - rightVal}
  204. case "*":
  205. return &object.Integer{Value: leftVal * rightVal}
  206. case "/":
  207. return &object.Integer{Value: leftVal / rightVal}
  208. case "<":
  209. return nativeBooleanObject(leftVal < rightVal)
  210. case ">":
  211. return nativeBooleanObject(leftVal > rightVal)
  212. case "==":
  213. return nativeBooleanObject(leftVal == rightVal)
  214. case "!=":
  215. return nativeBooleanObject(leftVal != rightVal)
  216. default:
  217. return newError("unknown operator: %s %s %s", left.Type(), operator, right.Type())
  218. }
  219. }
  220. func evalStringInfixExpression(operator string,
  221. left object.Object, right object.Object) object.Object {
  222. // Only identifier + is valid
  223. if operator != "+" {
  224. return newError("unknown operator: %s %s %s", left.Type(), operator, right.Type())
  225. }
  226. leftVal := left.(*object.String).Value
  227. rightVal := right.(*object.String).Value
  228. return &object.String{Value: leftVal + rightVal}
  229. }
  230. func evalMinusPreOperatorExpression(right object.Object) object.Object {
  231. if right.Type() != object.IntegerObj {
  232. return newError("unknown operator: -%s", right.Type())
  233. }
  234. value := right.(*object.Integer).Value
  235. return &object.Integer{Value: -value}
  236. }
  237. func evalBangOperatorExpression(right object.Object) object.Object {
  238. switch right {
  239. case TRUE:
  240. return FALSE
  241. case FALSE:
  242. return TRUE
  243. default:
  244. return FALSE
  245. }
  246. }
  247. func evalIdentifier(node *ast.Identifier, env *object.Environment) object.Object {
  248. if val, ok := env.Get(node.Value); ok {
  249. return val
  250. }
  251. if builtin, ok := builtins[node.Value]; ok {
  252. return builtin
  253. }
  254. return newError("identifier not found: " + node.Value)
  255. }
  256. func evalExpressions(exps []ast.Expression, env *object.Environment) []object.Object {
  257. var result []object.Object
  258. for _, e := range exps {
  259. evaluated := Eval(e, env)
  260. if isError(evaluated) {
  261. return []object.Object{evaluated}
  262. }
  263. result = append(result, evaluated)
  264. }
  265. return result
  266. }
  267. func evalIndexExpression(left, index object.Object) object.Object {
  268. switch {
  269. case left.Type() == object.ArrayObj && index.Type() == object.IntegerObj:
  270. return evalArrayIndexExpression(left, index)
  271. default:
  272. return newError("index operator not supported: %s", left.Type())
  273. }
  274. }
  275. func evalArrayIndexExpression(array, index object.Object) object.Object {
  276. arrayObj := array.(*object.Array)
  277. idx := index.(*object.Integer).Value
  278. max := int64(len(arrayObj.Elements) - 1)
  279. if idx < 0 || idx > max {
  280. return NULL
  281. }
  282. return arrayObj.Elements[idx]
  283. }
  284. func isTruthy(obj object.Object) bool {
  285. switch obj {
  286. case NULL:
  287. return false
  288. case TRUE:
  289. return true
  290. case FALSE:
  291. return false
  292. default:
  293. return true
  294. }
  295. }
  296. func isError(obj object.Object) bool {
  297. if obj != nil {
  298. return obj.Type() == object.ErrorObj
  299. }
  300. return false
  301. }