DISABLE FDs (REMOVE ME).
[jackhill/mal.git] / es6 / step4_if_fn_do.mjs
CommitLineData
e10ceff5
JM
1import rl from './node_readline.js'
2const readline = rl.readline
a05c086f 3import { _list_Q } from './types'
4f8c7db9
JM
4import { BlankException, read_str } from './reader'
5import { pr_str } from './printer'
6import { new_env, env_set, env_get } from './env'
7import { core_ns } from './core'
5024b694
JM
8
9// read
a05c086f 10const READ = str => read_str(str)
5024b694
JM
11
12// eval
13const eval_ast = (ast, env) => {
a05c086f 14 if (typeof ast === 'symbol') {
5024b694 15 return env_get(env, ast)
a05c086f
JM
16 } else if (ast instanceof Array) {
17 return ast.map(x => EVAL(x, env))
18 } else if (ast instanceof Map) {
4f8c7db9 19 let new_hm = new Map()
a05c086f 20 ast.forEach((v, k) => new_hm.set(EVAL(k, env), EVAL(v, env)))
4f8c7db9 21 return new_hm
5024b694 22 } else {
4f8c7db9 23 return ast
5024b694
JM
24 }
25}
26
27const EVAL = (ast, env) => {
4f8c7db9 28 //console.log('EVAL:', pr_str(ast, true))
5024b694 29 if (!_list_Q(ast)) { return eval_ast(ast, env) }
f8665761 30 if (ast.length === 0) { return ast }
5024b694 31
4f8c7db9 32 const [a0, a1, a2, a3] = ast
a05c086f
JM
33 switch (typeof a0 === 'symbol' ? Symbol.keyFor(a0) : Symbol(':default')) {
34 case 'def!':
4f8c7db9 35 return env_set(env, a1, EVAL(a2, env))
5024b694 36 case 'let*':
4f8c7db9 37 let let_env = new_env(env)
5024b694 38 for (let i=0; i < a1.length; i+=2) {
4f8c7db9 39 env_set(let_env, a1[i], EVAL(a1[i+1], let_env))
5024b694 40 }
4f8c7db9
JM
41 return EVAL(a2, let_env)
42 case 'do':
43 return eval_ast(ast.slice(1), env)[ast.length-2]
44 case 'if':
45 let cond = EVAL(a1, env)
5024b694 46 if (cond === null || cond === false) {
4f8c7db9 47 return typeof a3 !== 'undefined' ? EVAL(a3, env) : null
5024b694 48 } else {
4f8c7db9 49 return EVAL(a2, env)
5024b694 50 }
4f8c7db9
JM
51 case 'fn*':
52 return (...args) => EVAL(a2, new_env(env, a1, args))
5024b694 53 default:
4f8c7db9
JM
54 let [f, ...args] = eval_ast(ast, env)
55 return f(...args)
5024b694
JM
56 }
57}
58
59// print
a05c086f 60const PRINT = exp => pr_str(exp, true)
5024b694
JM
61
62// repl
4f8c7db9 63let repl_env = new_env()
a05c086f 64const REP = str => PRINT(EVAL(READ(str), repl_env))
5024b694
JM
65
66// core.EXT: defined using ES6
a05c086f 67for (let [k, v] of core_ns) { env_set(repl_env, Symbol.for(k), v) }
5024b694
JM
68
69// core.mal: defined using language itself
4f8c7db9 70REP('(def! not (fn* (a) (if a false true)))')
5024b694
JM
71
72while (true) {
4f8c7db9
JM
73 let line = readline('user> ')
74 if (line == null) break
5024b694 75 try {
a05c086f 76 if (line) { console.log(REP(line)) }
5024b694 77 } catch (exc) {
a05c086f 78 if (exc instanceof BlankException) { continue }
dd7a4f55
JM
79 if (exc instanceof Error) { console.warn(exc.stack) }
80 else { console.warn(`Error: ${exc}`) }
5024b694
JM
81 }
82}