evaluator.go 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247
  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. }
  72. return nil
  73. }
  74. func newError(format string, a ...interface{}) *object.Error {
  75. return &object.Error{Msg: fmt.Sprintf(format, a...)}
  76. }
  77. func evalProgram(node *ast.Program, env *object.Environment) object.Object {
  78. var result object.Object
  79. for _, stmt := range node.Statements {
  80. result = Eval(stmt, env)
  81. // skip return 后面的语句
  82. switch result := result.(type) {
  83. case *object.ReturnValue:
  84. return result.Value
  85. case *object.Error:
  86. return result
  87. }
  88. }
  89. return result
  90. }
  91. func nativeBooleanObject(input bool) *object.Boolean {
  92. if input {
  93. return TRUE
  94. } else {
  95. return FALSE
  96. }
  97. }
  98. func evalPrefixExpression(operator string, right object.Object) object.Object {
  99. switch operator {
  100. case "!":
  101. return evalBangOperatorExpression(right)
  102. case "-":
  103. return evalMinusPreOperatorExpression(right)
  104. default:
  105. return newError("unknown operator: %s%s", operator, right.Type())
  106. }
  107. }
  108. func evalInfixExpression(operator string, left object.Object, right object.Object) object.Object {
  109. switch {
  110. case left.Type() == object.IntegerObj && right.Type() == object.IntegerObj:
  111. return evalIntegerInfixExpression(operator, left, right)
  112. case operator == "==":
  113. return nativeBooleanObject(left == right)
  114. case operator == "!=":
  115. return nativeBooleanObject(left != right)
  116. case left.Type() != right.Type():
  117. return newError("type mismatch: %s %s %s", left.Type(), operator, right.Type())
  118. default:
  119. return newError("unknown operator: %s %s %s", left.Type(), operator, right.Type())
  120. }
  121. }
  122. func evalIfExpression(node *ast.IfExpression, env *object.Environment) object.Object {
  123. condition := Eval(node.Condition, env)
  124. if isError(condition) {
  125. return condition
  126. }
  127. if isTruthy(condition) {
  128. return Eval(node.Consequence, env)
  129. } else if node.Alternative != nil {
  130. return Eval(node.Alternative, env)
  131. } else {
  132. return NULL
  133. }
  134. }
  135. func evalBlockStatements(block *ast.BlockStatement, env *object.Environment) object.Object {
  136. var result object.Object
  137. for _, stmt := range block.Statements {
  138. result = Eval(stmt, env)
  139. if result != nil {
  140. rt := result.Type()
  141. if rt == object.ReturnValueObj || rt == object.ErrorObj {
  142. return result
  143. }
  144. }
  145. }
  146. return result
  147. }
  148. func evalIntegerInfixExpression(operator string,
  149. left object.Object, right object.Object) object.Object {
  150. leftVal := left.(*object.Integer).Value
  151. rightVal := right.(*object.Integer).Value
  152. switch operator {
  153. case "+":
  154. return &object.Integer{Value: leftVal + rightVal}
  155. case "-":
  156. return &object.Integer{Value: leftVal - rightVal}
  157. case "*":
  158. return &object.Integer{Value: leftVal * rightVal}
  159. case "/":
  160. return &object.Integer{Value: leftVal / rightVal}
  161. case "<":
  162. return nativeBooleanObject(leftVal < rightVal)
  163. case ">":
  164. return nativeBooleanObject(leftVal > rightVal)
  165. case "==":
  166. return nativeBooleanObject(leftVal == rightVal)
  167. case "!=":
  168. return nativeBooleanObject(leftVal != rightVal)
  169. default:
  170. return newError("unknown operator: %s %s %s", left.Type(), operator, right.Type())
  171. }
  172. }
  173. func evalMinusPreOperatorExpression(right object.Object) object.Object {
  174. if right.Type() != object.IntegerObj {
  175. return newError("unknown operator: -%s", right.Type())
  176. }
  177. value := right.(*object.Integer).Value
  178. return &object.Integer{Value: -value}
  179. }
  180. func evalBangOperatorExpression(right object.Object) object.Object {
  181. switch right {
  182. case TRUE:
  183. return FALSE
  184. case FALSE:
  185. return TRUE
  186. default:
  187. return FALSE
  188. }
  189. }
  190. func evalIdentifier(node *ast.Identifier, env *object.Environment) object.Object {
  191. val, ok := env.Get(node.Value)
  192. if !ok {
  193. return newError("identifier not found: " + node.Value)
  194. }
  195. return val
  196. }
  197. func evalExpressions(exps []ast.Expression, env *object.Environment) []object.Object {
  198. var result []object.Object
  199. for _, e := range exps {
  200. evaluated := Eval(e, env)
  201. if isError(evaluated) {
  202. return []object.Object{evaluated}
  203. }
  204. result = append(result, evaluated)
  205. }
  206. return result
  207. }
  208. func isTruthy(obj object.Object) bool {
  209. switch obj {
  210. case NULL:
  211. return false
  212. case TRUE:
  213. return true
  214. case FALSE:
  215. return false
  216. default:
  217. return true
  218. }
  219. }
  220. func isError(obj object.Object) bool {
  221. if obj != nil {
  222. return obj.Type() == object.ErrorObj
  223. }
  224. return false
  225. }