1
1
mirror of https://github.com/kanaka/mal.git synced 2024-11-13 11:23:59 +03:00
mal/fsharp/step2_eval.fs

72 lines
1.9 KiB
Forth

module REPL
open System
open Node
open Types
let rec eval_ast env = function
| Symbol(sym) -> Env.get env sym
| List(_, lst) -> lst |> List.map (eval env) |> makeList
| Vector(_, seg) -> seg |> Seq.map (eval env) |> Array.ofSeq |> Node.ofArray
| Map(_, map) -> map |> Map.map (fun k v -> eval env v) |> makeMap
| node -> node
and eval env = function
| List(_, []) as emptyList -> emptyList
| List(_, _) as node ->
let resolved = node |> eval_ast env
match resolved with
| List(_, BuiltInFunc(_, _, f)::rest) -> f rest
| _ -> raise <| Error.errExpectedX "func"
| node -> node |> eval_ast env
let READ input =
try
Reader.read_str input
with
| Error.ReaderError(msg) ->
printfn "%s" msg
[]
let EVAL env ast =
try
Some(eval env ast)
with
| Error.EvalError(str)
| Error.ReaderError(str) ->
printfn "%s" str
None
| Error.MalError(node) ->
printfn "%s" (Printer.pr_str [node])
None
| ex ->
printfn "%s" (ex.Message)
None
let PRINT v =
v
|> Seq.singleton
|> Printer.pr_str
|> printfn "%s"
let REP env input =
READ input
|> Seq.ofList
|> Seq.choose (fun form -> EVAL env form)
|> Seq.iter (fun value -> PRINT value)
let getReadlineMode args =
if args |> Array.exists (fun e -> e = "--raw") then
Readline.Mode.Raw
else
Readline.Mode.Terminal
[<EntryPoint>]
let rec main args =
let mode = getReadlineMode args
let env = Env.makeRootEnv ()
match Readline.read "user> " mode with
| null -> 0
| input ->
REP env input
main args