1
1
mirror of https://github.com/kanaka/mal.git synced 2024-09-20 01:57:09 +03:00
mal/haxe/Step3_env.hx
Joel Martin dd7a4f55f3 Test uncaught throw, catchless try* . Fix 46 impls.
Fixes made to: ada, c, chuck, clojure, coffee, common-lisp, cpp,
crystal, d, dart, elm, erlang, es6, factor, fsharp, gnu-smalltalk,
groovy, guile, haxe, hy, js, livescript, matlab, miniMAL, nasm, nim,
objc, objpascal, ocaml, perl, perl6, php, plsql, ps, python, r,
rpython, ruby, scheme, swift3, tcl, ts, vb, vimscript, wasm, yorick.

Catchless try* test is an optional test. Not all implementations
support catchless try* but a number were fixed so they at least don't
crash on catchless try*.
2018-12-12 14:18:26 -06:00

104 lines
3.2 KiB
Haxe

import Compat;
import types.Types.MalType;
import types.Types.*;
import reader.*;
import printer.*;
import env.*;
class Step3_env {
// READ
static function READ(str:String):MalType {
return Reader.read_str(str);
}
// EVAL
static function eval_ast(ast:MalType, env:Env) {
return switch (ast) {
case MalSymbol(s): env.get(ast);
case MalList(l):
MalList(l.map(function(x) { return EVAL(x, env); }));
case MalVector(l):
MalVector(l.map(function(x) { return EVAL(x, env); }));
case MalHashMap(m):
var new_map = new Map<String,MalType>();
for (k in m.keys()) {
new_map[k] = EVAL(m[k], env);
}
MalHashMap(new_map);
case _: ast;
}
}
static function EVAL(ast:MalType, env:Env):MalType {
if (!list_Q(ast)) { return eval_ast(ast, env); }
// apply
var alst = switch (ast) { case MalList(lst): lst; case _: []; }
if (alst.length == 0) { return ast; }
switch (alst[0]) {
case MalSymbol("def!"):
return env.set(alst[1], EVAL(alst[2], env));
case MalSymbol("let*"):
var let_env = new Env(env);
switch (alst[1]) {
case MalList(l) | MalVector(l):
for (i in 0...l.length) {
if ((i%2) > 0) { continue; }
let_env.set(l[i], EVAL(l[i+1], let_env));
}
case _: throw "Invalid let*";
}
return EVAL(alst[2], let_env);
case _:
var el = eval_ast(ast, env);
var lst = _list(el);
switch (first(el)) {
case MalFunc(f,_,_,_,_,_): return f(_list(el).slice(1));
case _: throw "Call of non-function";
}
}
}
// PRINT
static function PRINT(exp:MalType):String {
return Printer.pr_str(exp, true);
}
// repl
static function NumOp(op):MalType {
return MalFunc(function(args:Array<MalType>) {
return switch (args) {
case [MalInt(a), MalInt(b)]: MalInt(op(a,b));
case _: throw "Invalid numeric op call";
}
},null,null,null,false,nil);
}
static var repl_env = new Env(null);
static function rep(line:String):String {
return PRINT(EVAL(READ(line), repl_env));
}
public static function main() {
repl_env.set(MalSymbol("+"), NumOp(function(a,b) {return a+b;}));
repl_env.set(MalSymbol("-"), NumOp(function(a,b) {return a-b;}));
repl_env.set(MalSymbol("*"), NumOp(function(a,b) {return a*b;}));
repl_env.set(MalSymbol("/"), NumOp(function(a,b) {return Std.int(a/b);}));
while (true) {
try {
var line = Compat.readline("user> ");
if (line == "") { continue; }
Compat.println(rep(line));
} catch (exc:BlankLine) {
continue;
} catch (exc:haxe.io.Eof) {
Compat.exit(0);
} catch (exc:Dynamic) {
Compat.println("Error: " + exc);
}
}
}
}