nomsu/nomic.moon

576 lines
21 KiB
Plaintext
Raw Normal View History

2017-09-05 23:51:35 -07:00
#!/usr/bin/env moon
re = require 're'
lpeg = require 'lpeg'
2017-08-18 21:21:24 -07:00
utils = require 'utils'
2017-09-05 23:51:35 -07:00
-- TODO:
-- string interpolation
2017-09-11 13:05:25 -07:00
-- comprehensions?
-- dicts?
-- better scoping?
-- first-class functions
2017-09-05 23:51:35 -07:00
2017-09-11 13:05:25 -07:00
INDENT = " "
2017-08-22 01:02:41 -07:00
lpeg.setmaxstack 10000 -- whoa
{:P,:V,:S,:Cg,:C,:Cp,:B,:Cmt} = lpeg
wordchar = P(1)-S(' \t\n\r%:;,.{}[]()"')
2017-09-07 09:38:54 -07:00
comment = re.compile [[comment <- "(#" (comment / ((! "#)") .))* "#)"]]
whitespace = (S(" \t") + comment)^1
2017-09-05 23:51:35 -07:00
nl = P("\n")
2017-09-07 09:38:54 -07:00
blank_line = whitespace^-1 * nl
2017-08-22 01:02:41 -07:00
get_line_indentation = (line)->
indent_amounts = {[" "]:1, ["\t"]:4}
with sum = 0
leading_space = line\gsub("([\t ]*).*", "%1")
for c in leading_space\gmatch "[\t ]"
sum += indent_amounts[c]
2017-09-05 23:51:35 -07:00
make_parser = (lingo, extra_definitions)->
indent_stack = {0}
push = (n)-> table.insert indent_stack, n
pop = ()-> table.remove indent_stack
check_indent = (subject,end_pos,spaces)->
num_spaces = get_line_indentation(spaces)
if num_spaces <= indent_stack[#indent_stack] then return nil
push num_spaces
return end_pos
check_dedent = (subject,end_pos,spaces)->
num_spaces = get_line_indentation(spaces)
if num_spaces >= indent_stack[#indent_stack] then return nil
pop!
return end_pos
check_nodent = (subject,end_pos,spaces)->
num_spaces = get_line_indentation(spaces)
if num_spaces != indent_stack[#indent_stack] then return nil
return end_pos
defs =
2017-09-07 09:38:54 -07:00
:wordchar, :nl, ws:whitespace, :comment
2017-09-05 23:51:35 -07:00
eol: #nl + (P("")-P(1))
2017-09-12 17:42:35 -07:00
word_boundary: whitespace + B(P("..")) + B(S("\";)]")) + #S("\":([") + #((whitespace + nl)^0 * P(".."))
2017-09-07 09:38:54 -07:00
indent: #(nl * blank_line^0 * Cmt(whitespace^-1, check_indent))
dedent: #(nl * blank_line^0 * Cmt(whitespace^-1, check_dedent))
new_line: nl * blank_line^0 * Cmt(whitespace^-1, check_nodent)
2017-09-05 23:51:35 -07:00
error_handler: (src,pos,errors)->
line_no = 1
for _ in src\sub(1,-#errors)\gmatch("\n") do line_no += 1
err_pos = #src - #errors + 1
if errors\sub(1,1) == "\n"
-- Indentation error
err_pos += #errors\match("[ \t]*", 2)
start_of_err_line = err_pos
while src\sub(start_of_err_line, start_of_err_line) != "\n" do start_of_err_line -= 1
start_of_prev_line = start_of_err_line - 1
while src\sub(start_of_prev_line, start_of_prev_line) != "\n" do start_of_prev_line -= 1
prev_line,err_line,next_line = src\match("([^\n]*)\n([^\n]*)\n([^\n]*)", start_of_prev_line+1)
2017-09-07 09:38:54 -07:00
pointer = ("-")\rep(err_pos - start_of_err_line + 0) .. "^"
error("\nParse error on line #{line_no}:\n\n#{prev_line}\n#{err_line}\n#{pointer}\n#{next_line}\n")
2017-09-05 23:51:35 -07:00
if extra_definitions
for k,v in pairs(extra_definitions) do defs[k] = v
setmetatable(defs, {
__index: (t,key)->
fn = (src, value, errors)->
token = {type: key, :src, :value, :errors}
return token
t[key] = fn
return fn
})
return re.compile lingo, defs
2017-09-12 20:00:19 -07:00
class Compiler
2017-08-22 02:52:05 -07:00
new:(parent)=>
@defs = setmetatable({}, {__index:parent and parent.defs})
2017-09-12 20:00:19 -07:00
@callstack = {}
2017-08-22 01:02:41 -07:00
@debug = false
2017-09-12 20:00:19 -07:00
@initialize_core!
2017-08-22 01:02:41 -07:00
call: (fn_name,...)=>
fn_info = @defs[fn_name]
if fn_info == nil
2017-09-12 21:10:22 -07:00
@error "Attempt to call undefined function: #{fn_name}"
if fn_info.is_macro
2017-09-12 21:10:22 -07:00
@error "Attempt to call macro at runtime: #{fn_name}"
2017-09-12 20:00:19 -07:00
unless @check_permission(fn_name)
2017-09-12 21:10:22 -07:00
@error "You do not have the authority to call: #{fn_name}"
2017-09-12 20:00:19 -07:00
table.insert @callstack, fn_name
{:fn, :arg_names} = fn_info
args = {name, select(i,...) for i,name in ipairs(arg_names)}
2017-08-18 17:08:15 -07:00
if @debug
print "Calling #{fn_name} with args: #{utils.repr(args)}"
2017-09-12 20:00:19 -07:00
ret = fn(self, args)
table.remove @callstack
return ret
check_permission: (fn_name)=>
fn_info = @defs[fn_name]
if fn_info == nil
2017-09-12 21:10:22 -07:00
@error "Undefined function: #{fn_name}"
2017-09-12 20:00:19 -07:00
if fn_info.whiteset == nil then return true
for caller in *@callstack
if fn_info.whiteset[caller]
return true
return false
2017-08-22 01:02:41 -07:00
def: (spec, fn)=>
2017-09-12 20:00:19 -07:00
if @debug
print "Defining rule: #{spec}"
2017-09-12 15:28:58 -07:00
invocations,arg_names = @get_invocations spec
fn_info = {:fn, :arg_names, :invocations, is_macro:false}
2017-08-22 01:02:41 -07:00
for invocation in *invocations
@defs[invocation] = fn_info
2017-08-18 17:08:15 -07:00
2017-08-22 01:02:41 -07:00
get_invocations:(text)=>
if type(text) == 'string' then text = {text}
invocations = {}
local arg_names
for _text in *text
2017-09-07 09:38:54 -07:00
invocation = _text\gsub("%%%S+","%%")
_arg_names = [arg for arg in _text\gmatch("%%(%S+)")]
2017-08-22 01:02:41 -07:00
table.insert(invocations, invocation)
2017-09-07 09:38:54 -07:00
if arg_names
if not utils.equivalent(utils.set(arg_names), utils.set(_arg_names))
2017-09-12 21:10:22 -07:00
@error("Conflicting argument names #{utils.repr(arg_names)} and #{utils.repr(_arg_names)} for #{utils.repr(text)}")
2017-09-07 09:38:54 -07:00
else arg_names = _arg_names
2017-08-22 01:02:41 -07:00
return invocations, arg_names
defmacro: (spec, lua_gen_fn)=>
2017-09-12 15:28:58 -07:00
invocations,arg_names = @get_invocations spec
fn_info = {fn:lua_gen_fn, :arg_names, :invocations, is_macro:true}
2017-08-22 01:02:41 -07:00
for invocation in *invocations
@defs[invocation] = fn_info
2017-08-22 01:02:41 -07:00
run: (text)=>
if @debug
2017-09-07 09:38:54 -07:00
print "RUNNING TEXT:\n#{text}"
2017-09-12 15:28:58 -07:00
code = @compile(text)
2017-08-22 01:02:41 -07:00
if @debug
2017-09-07 09:38:54 -07:00
print "\nGENERATED LUA CODE:\n#{code}"
2017-09-12 20:00:19 -07:00
[==[
2017-08-22 01:02:41 -07:00
lua_thunk, err = loadstring(code)
2017-08-18 17:08:15 -07:00
if not lua_thunk
2017-08-22 04:15:13 -07:00
error("Failed to compile generated code:\n#{code}\n\n#{err}")
2017-08-22 01:02:41 -07:00
action = lua_thunk!
if @debug
print("Running...")
return action(self, {})
2017-09-12 20:00:19 -07:00
]==]
code
2017-08-22 01:02:41 -07:00
parse: (str)=>
if @debug
print("PARSING:\n#{str}")
lingo = [=[
2017-09-12 17:42:35 -07:00
file <- ({ {| %ws? %new_line? {:body: block :} %new_line? %ws? (errors)? |} }) -> File
errors <- (({.+}) => error_handler)
block <- ({ {| statement (%new_line statement)* |} }) -> Block
statement <- ({ (functioncall / expression) }) -> Statement
one_liner <- ({ {|
(({
(({ {|
(expression (%word_boundary fn_bit)+) / (word (%word_boundary fn_bit)*)
|} }) -> FunctionCall)
/ (expression)
}) -> Statement)
|} }) -> Block
functioncall <- ({ {| (expression %word_boundary fn_bits) / (word (%word_boundary fn_bits)?) |} }) -> FunctionCall
fn_bit <- (expression / word)
fn_bits <-
((".." %ws? (%indent %new_line indented_fn_bits %dedent) (%new_line ".." %ws? fn_bits)?)
2017-09-12 17:42:35 -07:00
/ (%new_line ".." fn_bit (%word_boundary fn_bits)?)
/ (fn_bit (%word_boundary fn_bits)?))
indented_fn_bits <-
fn_bit ((%new_line / %word_boundary) indented_fn_bits)?
thunk <-
({ ":" %ws?
((%indent %new_line block ((%dedent (%new_line "..")?) / errors))
/ (one_liner (%ws? (%new_line? ".."))?)) }) -> Thunk
word <- ({ !number {%wordchar+} }) -> Word
expression <- ({ (longstring / string / number / variable / list / thunk / subexpression) }) -> Expression
2017-09-12 15:28:58 -07:00
string <- ({ (!longstring) '"' {(("\" .) / [^"])*} '"' }) -> String
2017-09-07 09:38:54 -07:00
longstring <- ({ '".."' %ws? %indent {(%new_line "|" [^%nl]*)+} ((%dedent (%new_line '..')?) / errors) }) -> Longstring
number <- ({ {'-'? [0-9]+ ("." [0-9]+)?} }) -> Number
variable <- ({ ("%" {%wordchar+}) }) -> Var
subexpression <-
2017-09-07 09:38:54 -07:00
(!%comment "(" %ws? (functioncall / expression) %ws? ")")
/ ("(..)" %ws? %indent %new_line ((({ {| indented_fn_bits |} }) -> FunctionCall) / expression) %dedent (%new_line "..")?)
list <- ({ {|
("[..]" %ws? %indent %new_line indented_list ","? ((%dedent (%new_line "..")?) / errors))
/ ("[" %ws? (list_items ","?)? %ws?"]")
|} }) -> List
2017-09-12 17:42:35 -07:00
list_items <- ((functioncall / expression) (list_sep list_items)?)
list_sep <- %ws? "," %ws?
indented_list <-
2017-09-12 17:42:35 -07:00
(functioncall / expression) (((list_sep %new_line?) / %new_line) indented_list)?
]=]
2017-09-05 23:51:35 -07:00
lingo = make_parser lingo
tree = lingo\match(str\gsub("\r","").."\n")
2017-08-22 01:02:41 -07:00
if @debug
print("\nPARSE TREE:")
2017-09-12 15:28:58 -07:00
@print_tree(tree)
2017-08-22 01:02:41 -07:00
assert tree, "Failed to parse: #{str}"
return tree
2017-09-11 13:05:25 -07:00
tree_to_value: (tree)=>
2017-09-12 20:00:19 -07:00
code = "return (function(compiler, vars)\nreturn #{@tree_to_lua(tree)}\nend)"
lua_thunk, err = load(code)
if not lua_thunk
error("Failed to compile generated code:\n#{code}\n\n#{err}")
return (lua_thunk!)(self, {})
2017-09-11 13:05:25 -07:00
tree_to_lua: (tree, kind="Expression")=>
assert tree, "No tree provided."
indent = ""
buffer = {}
to_lua = (t,kind)->
2017-09-12 15:28:58 -07:00
ret = @tree_to_lua(t,kind)
2017-09-11 13:05:25 -07:00
return ret
add = (code)-> table.insert(buffer, code)
2017-08-22 01:02:41 -07:00
switch tree.type
when "File"
2017-09-12 20:00:19 -07:00
add [[return (function(compiler, vars)
2017-09-11 13:05:25 -07:00
local ret]]
2017-09-12 20:00:19 -07:00
vars = {}
for statement in *tree.value.body.value
code = to_lua(statement)
-- Run the fuckers as we go
2017-09-12 21:10:22 -07:00
lua_thunk, err = load("return (function(compiler, vars)\n#{code}\nend)")
2017-09-12 20:00:19 -07:00
if not lua_thunk
error("Failed to compile generated code:\n#{code}\n\n#{err}")
2017-09-12 21:10:22 -07:00
ok,err = pcall(lua_thunk)
if not ok then error(err)
ok,err = pcall(err, self, vars)
if not ok then @error(err)
2017-09-12 20:00:19 -07:00
add code
2017-09-11 13:05:25 -07:00
add [[
return ret
end)
]]
2017-08-22 01:02:41 -07:00
when "Block"
2017-09-12 20:00:19 -07:00
for statement in *tree.value
add to_lua(statement)
2017-08-22 01:02:41 -07:00
when "Thunk"
2017-09-11 13:05:25 -07:00
assert tree.value.type == "Block", "Non-block value in Thunk"
add [[
2017-09-12 20:00:19 -07:00
(function(compiler, vars)
2017-09-11 13:05:25 -07:00
local ret]]
add to_lua(tree.value)
add [[
return ret
end)
]]
2017-08-22 01:02:41 -07:00
when "Statement"
2017-09-11 13:05:25 -07:00
-- This case here is to prevent "ret =" from getting prepended when the macro might not want it
2017-08-22 04:15:13 -07:00
if tree.value.type == "FunctionCall"
2017-09-12 15:28:58 -07:00
name = @fn_name_from_tree(tree.value)
if @defs[name] and @defs[name].is_macro
2017-09-11 13:05:25 -07:00
add @run_macro(tree.value, "Statement")
else
add "ret = "..(to_lua(tree.value)\match("%s*(.*)"))
else
add "ret = "..(to_lua(tree.value)\match("%s*(.*)"))
2017-08-22 01:02:41 -07:00
when "Expression"
2017-09-11 13:05:25 -07:00
add to_lua(tree.value)
2017-08-22 01:02:41 -07:00
when "FunctionCall"
2017-09-12 15:28:58 -07:00
name = @fn_name_from_tree(tree)
if @defs[name] and @defs[name].is_macro
2017-09-11 13:05:25 -07:00
add @run_macro(tree, "Expression")
2017-08-22 01:02:41 -07:00
else
2017-09-11 13:05:25 -07:00
args = [to_lua(a) for a in *tree.value when a.type != "Word"]
2017-08-22 01:02:41 -07:00
table.insert args, 1, utils.repr(name, true)
2017-09-12 20:00:19 -07:00
add @@comma_separated_items("compiler:call(", args, ")")
2017-08-22 01:02:41 -07:00
when "String"
2017-08-23 14:02:36 -07:00
escapes = n:"\n", t:"\t", b:"\b", a:"\a", v:"\v", f:"\f", r:"\r"
unescaped = tree.value\gsub("\\(.)", ((c)-> escapes[c] or c))
2017-09-11 13:05:25 -07:00
add utils.repr(unescaped, true)
2017-08-22 01:02:41 -07:00
when "Longstring"
2017-09-11 13:05:25 -07:00
-- TODO: handle comments here?
2017-09-07 09:38:54 -07:00
result = [line for line in tree.value\gmatch("[ \t]*|([^\n]*)")]
2017-09-11 13:05:25 -07:00
add utils.repr(table.concat(result, "\n"), true)
2017-08-22 01:02:41 -07:00
when "Number"
2017-09-11 13:05:25 -07:00
add tree.value
2017-08-22 01:02:41 -07:00
when "List"
if #tree.value == 0
2017-09-11 13:05:25 -07:00
add "{}"
2017-08-22 01:02:41 -07:00
elseif #tree.value == 1
2017-09-12 15:28:58 -07:00
add "{#{to_lua(tree.value[1])}}"
2017-08-22 01:02:41 -07:00
else
2017-09-11 13:05:25 -07:00
add @@comma_separated_items("{", [to_lua(item) for item in *tree.value], "}")
2017-08-22 01:02:41 -07:00
when "Var"
2017-09-11 13:05:25 -07:00
add "vars[#{utils.repr(tree.value,true)}]"
2017-08-22 01:02:41 -07:00
else
error("Unknown/unimplemented thingy: #{tree.type}")
2017-09-11 13:05:25 -07:00
-- TODO: make indentation clean
2017-09-11 13:33:15 -07:00
buffer = table.concat(buffer, "\n")
return buffer
2017-09-11 13:05:25 -07:00
@comma_separated_items: (open, items, close)=>
utils.accumulate "\n", ->
buffer = open
so_far = 0
for i,item in ipairs(items)
if i < #items then item ..= ", "
if so_far + #item >= 80 and #buffer > 0
coroutine.yield buffer
so_far -= #buffer
buffer = item
else
so_far += #item
buffer ..= item
buffer ..= close
coroutine.yield buffer
2017-09-12 15:28:58 -07:00
fn_name_from_tree: (tree)=>
assert(tree.type == "FunctionCall", "Attempt to get fn name from non-functioncall tree: #{tree.type}")
2017-09-11 13:05:25 -07:00
name_bits = {}
for token in *tree.value
table.insert name_bits, if token.type == "Word" then token.value else "%"
2017-09-12 15:28:58 -07:00
table.concat(name_bits, " ")
run_macro: (tree, kind="Expression")=>
name = @fn_name_from_tree(tree)
2017-09-11 13:05:25 -07:00
unless @defs[name] and @defs[name].is_macro
2017-09-12 21:10:22 -07:00
@error("Macro not found: #{name}")
2017-09-12 20:00:19 -07:00
unless @check_permission(name)
2017-09-12 21:10:22 -07:00
@error "You do not have the authority to call: #{name}"
2017-09-11 13:05:25 -07:00
{:fn, :arg_names} = @defs[name]
args = [a for a in *tree.value when a.type != "Word"]
args = {name,args[i] for i,name in ipairs(arg_names)}
2017-09-12 20:00:19 -07:00
table.insert @callstack, name
2017-09-11 13:05:25 -07:00
ret, manual_mode = fn(self, args, kind)
2017-09-12 20:00:19 -07:00
table.remove @callstack
2017-09-11 13:05:25 -07:00
if not ret
2017-09-12 21:10:22 -07:00
@error("No return value for macro: #{name}")
2017-09-11 13:05:25 -07:00
if kind == "Statement" and not manual_mode
ret = "ret = "..ret
2017-08-18 17:08:15 -07:00
return ret
2017-08-22 01:02:41 -07:00
_yield_tree: (tree, indent_level=0)=>
2017-09-11 13:05:25 -07:00
ind = (s) -> INDENT\rep(indent_level)..s
2017-08-22 01:02:41 -07:00
switch tree.type
when "File"
coroutine.yield(ind"File:")
2017-09-12 15:28:58 -07:00
@_yield_tree(tree.value.body, indent_level+1)
2017-08-22 01:02:41 -07:00
when "Errors"
coroutine.yield(ind"Error:\n#{tree.value}")
when "Block"
for chunk in *tree.value
2017-09-12 15:28:58 -07:00
@_yield_tree(chunk, indent_level)
2017-08-22 01:02:41 -07:00
when "Thunk"
coroutine.yield(ind"Thunk:")
2017-09-12 15:28:58 -07:00
@_yield_tree(tree.value, indent_level+1)
2017-08-22 01:02:41 -07:00
when "Statement"
2017-09-12 15:28:58 -07:00
@_yield_tree(tree.value, indent_level)
2017-08-22 01:02:41 -07:00
when "Expression"
2017-09-12 15:28:58 -07:00
@_yield_tree(tree.value, indent_level)
2017-08-22 01:02:41 -07:00
when "FunctionCall"
2017-09-12 15:28:58 -07:00
name = @fn_name_from_tree(tree)
args = [a for a in *tree.value when a.type != "Word"]
if #args == 0
2017-08-22 01:02:41 -07:00
coroutine.yield(ind"Call [#{name}]!")
else
coroutine.yield(ind"Call [#{name}]:")
2017-09-12 15:28:58 -07:00
for a in *args
@_yield_tree(a, indent_level+1)
2017-08-22 01:02:41 -07:00
when "String"
-- TODO: Better implement
coroutine.yield(ind(utils.repr(tree.value, true)))
when "Longstring"
-- TODO: Better implement
2017-08-22 01:02:41 -07:00
coroutine.yield(ind(utils.repr(tree.value, true)))
when "Number"
coroutine.yield(ind(tree.value))
when "List"
if #tree.value == 0
coroutine.yield(ind("<Empty List>"))
else
coroutine.yield(ind"List:")
for item in *tree.value
2017-09-12 15:28:58 -07:00
@_yield_tree(item, indent_level+1)
2017-08-22 01:02:41 -07:00
when "Var"
coroutine.yield ind"Var[#{utils.repr(tree.value)}]"
else
error("Unknown/unimplemented thingy: #{tree.type}")
return nil -- to prevent tail calls
print_tree:(tree)=>
2017-09-12 15:28:58 -07:00
for line in coroutine.wrap(-> @_yield_tree(tree))
2017-08-22 01:02:41 -07:00
print(line)
stringify_tree:(tree)=>
result = {}
2017-09-12 15:28:58 -07:00
for line in coroutine.wrap(-> @_yield_tree(tree))
2017-08-22 01:02:41 -07:00
table.insert(result, line)
return table.concat result, "\n"
2017-09-12 20:00:19 -07:00
compile: (src, output_file=nil)=>
if @debug
print "COMPILING:\n#{src}"
2017-09-12 15:28:58 -07:00
tree = @parse(src)
2017-09-11 13:05:25 -07:00
assert tree, "Tree failed to compile: #{src}"
2017-09-12 15:28:58 -07:00
code = @tree_to_lua(tree)
2017-09-12 20:00:19 -07:00
if output_file
output = io.open(output_file, "w")
output\write(code)
2017-08-22 01:02:41 -07:00
return code
2017-09-12 21:10:22 -07:00
error: (...)=>
print(...)
print("Callstack:")
for i=#@callstack,1,-1
print " #{@callstack[i]}"
error!
2017-08-22 01:02:41 -07:00
test: (src, expected)=>
i = 1
while i != nil
start,stop = src\find("\n\n", i)
test = src\sub(i,start)
i = stop
start,stop = test\find"==="
2017-08-22 01:02:41 -07:00
if not start or not stop then
2017-09-12 21:10:22 -07:00
@error("WHERE'S THE ===? in:\n#{test}")
test_src, expected = test\sub(1,start-1), test\sub(stop+1,-1)
expected = expected\match'[\n]*(.*[^\n])'
2017-09-12 15:28:58 -07:00
tree = @parse(test_src)
got = @stringify_tree(tree.value.body)
if got != expected
2017-09-12 21:10:22 -07:00
@error"TEST FAILED!\nSource:\n#{test_src}\nExpected:\n#{expected}\n\nGot:\n#{got}"
2017-08-18 17:08:15 -07:00
2017-08-18 17:06:47 -07:00
2017-09-12 20:00:19 -07:00
initialize_core: =>
-- Sets up some core functionality
as_lua_code = (str)=>
switch str.type
when "String"
escapes = n:"\n", t:"\t", b:"\b", a:"\a", v:"\v", f:"\f", r:"\r"
unescaped = str.value\gsub("\\(.)", ((c)-> escapes[c] or c))
return unescaped
when "Longstring"
-- TODO: handle comments?
result = [line for line in str.value\gmatch("[ \t]*|([^\n]*)")]
return table.concat(result, "\n")
else
return @tree_to_lua(str)
@defmacro [[lua block %lua_code]], (vars, kind)=>
if kind == "Expression" then error("Expected to be in statement.")
lua_code = vars.lua_code.value
switch lua_code.type
when "List"
-- TODO: handle subexpressions
return table.concat([as_lua_code(@, i.value) for i in *lua_code.value]), true
else
return as_lua_code(@, lua_code), true
@defmacro [[lua expr %lua_code]], (vars, kind)=>
lua_code = vars.lua_code.value
switch lua_code.type
when "List"
-- TODO: handle subexpressions
return table.concat([as_lua_code(@, i.value) for i in *lua_code.value])
else
return as_lua_code(@, lua_code)
@def "rule %spec %body", (vars)=>
@def vars.spec, vars.body
@defmacro [[macro %spec %body]], (vars, kind)=>
if kind == "Expression"
error("Macro definitions cannot be used as expressions.")
@defmacro @tree_to_value(vars.spec), @tree_to_value(vars.body)
return "", true
@defmacro [[macro block %spec %body]], (vars, kind)=>
if kind == "Expression"
error("Macro definitions cannot be used as expressions.")
invocation = @tree_to_value(vars.spec)
fn = @tree_to_value(vars.body)
@defmacro invocation, ((vars,kind)=>
if kind == "Expression"
error("Macro: #{invocation} was defined to be a block, not an expression.")
return fn(@,vars,kind), true)
return "", true
@def "run file %filename", (vars)=>
file = io.open(vars.filename)
return @run(file\read('*a'))
-- Run on the command line via "./nomic.moon input_file.nom" to execute
-- and "./nomic.moon input_file.nom output_file.lua" to compile (use "-" to compile to stdout)
if arg[1]
c = Compiler()
input = io.open(arg[1])\read("*a")
-- Kinda hacky, if run via "./nomic.moon file.nom -", then silence print and io.write
-- during execution and re-enable them to print out the generated source code
_print = print
_io_write = io.write
if arg[2] == "-"
export print
nop = ->
print, io.write = nop, nop
code = c\run(input)
if arg[2]
output = if arg[2] == "-"
export print
print, io.write = _print, _io_write
io.output()
else io.open(arg[2], 'w')
output\write [[
local load = function()
]]
output\write(code)
output\write [[
end
local utils = require('utils')
local Compiler = require('nomic')
local c = Compiler(require('core'))
load()(c, {})
]]
return Compiler