2014-11-09 01:56:36 +03:00
|
|
|
readline = require "./node_readline.coffee"
|
|
|
|
types = require "./types.coffee"
|
|
|
|
reader = require "./reader.coffee"
|
|
|
|
printer = require "./printer.coffee"
|
|
|
|
Env = require("./env.coffee").Env
|
|
|
|
core = require("./core.coffee")
|
|
|
|
|
|
|
|
# read
|
|
|
|
READ = (str) -> reader.read_str str
|
|
|
|
|
|
|
|
# eval
|
|
|
|
eval_ast = (ast, env) ->
|
2014-12-19 05:33:49 +03:00
|
|
|
if types._symbol_Q(ast) then env.get ast
|
2014-11-09 01:56:36 +03:00
|
|
|
else if types._list_Q(ast) then ast.map((a) -> EVAL(a, env))
|
|
|
|
else if types._vector_Q(ast)
|
|
|
|
types._vector(ast.map((a) -> EVAL(a, env))...)
|
|
|
|
else if types._hash_map_Q(ast)
|
|
|
|
new_hm = {}
|
|
|
|
new_hm[k] = EVAL(ast[k],env) for k,v of ast
|
|
|
|
new_hm
|
|
|
|
else ast
|
|
|
|
|
|
|
|
EVAL = (ast, env) ->
|
|
|
|
#console.log "EVAL:", printer._pr_str ast
|
|
|
|
if !types._list_Q ast then return eval_ast ast, env
|
2016-03-28 05:04:55 +03:00
|
|
|
if ast.length == 0 then return ast
|
2014-11-09 01:56:36 +03:00
|
|
|
|
|
|
|
# apply list
|
|
|
|
[a0, a1, a2, a3] = ast
|
|
|
|
switch a0.name
|
|
|
|
when "def!"
|
2014-12-19 05:33:49 +03:00
|
|
|
env.set(a1, EVAL(a2, env))
|
2014-11-09 01:56:36 +03:00
|
|
|
when "let*"
|
|
|
|
let_env = new Env(env)
|
|
|
|
for k,i in a1 when i %% 2 == 0
|
2014-12-19 05:33:49 +03:00
|
|
|
let_env.set(a1[i], EVAL(a1[i+1], let_env))
|
2014-11-09 01:56:36 +03:00
|
|
|
EVAL(a2, let_env)
|
|
|
|
when "do"
|
|
|
|
el = eval_ast(ast[1..], env)
|
|
|
|
el[el.length-1]
|
|
|
|
when "if"
|
|
|
|
cond = EVAL(a1, env)
|
|
|
|
if cond == null or cond == false
|
|
|
|
if a3? then EVAL(a3, env) else null
|
|
|
|
else
|
|
|
|
EVAL(a2, env)
|
|
|
|
when "fn*"
|
|
|
|
(args...) -> EVAL(a2, new Env(env, a1, args))
|
|
|
|
else
|
|
|
|
[f, args...] = eval_ast ast, env
|
|
|
|
f(args...)
|
|
|
|
|
|
|
|
|
|
|
|
# print
|
|
|
|
PRINT = (exp) -> printer._pr_str exp, true
|
|
|
|
|
|
|
|
# repl
|
|
|
|
repl_env = new Env()
|
|
|
|
rep = (str) -> PRINT(EVAL(READ(str), repl_env))
|
|
|
|
|
|
|
|
# core.coffee: defined using CoffeeScript
|
2014-12-19 05:33:49 +03:00
|
|
|
repl_env.set types._symbol(k), v for k,v of core.ns
|
2014-11-09 01:56:36 +03:00
|
|
|
|
|
|
|
# core.mal: defined using the language itself
|
|
|
|
rep("(def! not (fn* (a) (if a false true)))");
|
|
|
|
|
|
|
|
# repl loop
|
|
|
|
while (line = readline.readline("user> ")) != null
|
|
|
|
continue if line == ""
|
|
|
|
try
|
|
|
|
console.log rep line
|
|
|
|
catch exc
|
|
|
|
continue if exc instanceof reader.BlankException
|
2015-10-08 23:15:25 +03:00
|
|
|
if exc.stack? and exc.stack.length > 2000
|
|
|
|
console.log exc.stack.slice(0,1000) + "\n ..." + exc.stack.slice(-1000)
|
|
|
|
else if exc.stack? console.log exc.stack
|
|
|
|
else console.log exc
|
2014-11-09 01:56:36 +03:00
|
|
|
|
|
|
|
# vim: ts=2:sw=2
|