2017-09-05 23:51:35 -07:00
|
|
|
#!/usr/bin/env moon
|
2017-08-16 04:35:35 -07:00
|
|
|
re = require 're'
|
|
|
|
lpeg = require 'lpeg'
|
2017-08-18 21:21:24 -07:00
|
|
|
utils = require 'utils'
|
2017-08-16 04:35:35 -07:00
|
|
|
|
2017-09-05 23:51:35 -07:00
|
|
|
-- TODO:
|
2017-09-13 16:22:04 -07:00
|
|
|
-- improve indentation of generated lua code
|
|
|
|
-- provide way to run precompiled nomsu -> lua code
|
2017-09-11 13:05:25 -07:00
|
|
|
-- better scoping?
|
|
|
|
-- first-class functions
|
2017-09-13 16:08:26 -07:00
|
|
|
-- better error reporting
|
|
|
|
-- type checking?
|
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
|
2017-09-05 21:20:11 -07:00
|
|
|
{:P,:V,:S,:Cg,:C,:Cp,:B,:Cmt} = lpeg
|
2017-08-16 04:35:35 -07:00
|
|
|
|
2017-08-22 01:02:41 -07:00
|
|
|
get_line_indentation = (line)->
|
|
|
|
indent_amounts = {[" "]:1, ["\t"]:4}
|
|
|
|
with sum = 0
|
2017-09-14 00:09:54 -07:00
|
|
|
leading_space = line\match("[\t ]*")
|
2017-08-22 01:02:41 -07:00
|
|
|
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
|
|
|
|
|
2017-09-14 02:41:10 -07:00
|
|
|
wordchar = P(1)-S(' \t\n\r%#:;,.{}[]()"\\')
|
2017-09-14 00:09:54 -07:00
|
|
|
nl = P("\n")
|
|
|
|
whitespace = S(" \t")^1
|
|
|
|
blank_line = whitespace^-1 * nl
|
|
|
|
line_comment = re.compile([=[ "#" [^%nl]* ]=], {:nl})
|
|
|
|
block_comment = re.compile([=[
|
|
|
|
"#.." (!%nl .)* (%indent (!%dedent %nl [^%nl]*)*)
|
|
|
|
]=], {:nl, :whitespace,
|
|
|
|
indent:#(nl * blank_line^0 * Cmt(S(" \t")^0, check_indent)),
|
|
|
|
dedent:#(nl * blank_line^0 * Cmt(S(" \t")^0, check_dedent)),
|
|
|
|
new_line:nl * blank_line^0 * Cmt(S(" \t")^0, check_nodent)})
|
|
|
|
blank_line = ((Cmt(whitespace^-1, check_nodent) * (block_comment + line_comment))^-1 + whitespace^-1) * nl
|
2017-09-05 23:51:35 -07:00
|
|
|
defs =
|
2017-09-14 00:09:54 -07:00
|
|
|
:wordchar, :nl, ws:whitespace, :blank_line, :block_comment, :line_comment
|
2017-09-05 23:51:35 -07:00
|
|
|
eol: #nl + (P("")-P(1))
|
2017-09-14 00:09:54 -07:00
|
|
|
word_boundary: whitespace^-1 + 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-13 16:22:04 -07:00
|
|
|
class NomsuCompiler
|
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,...)=>
|
2017-09-05 21:20:11 -07:00
|
|
|
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}"
|
2017-09-11 19:23:55 -07:00
|
|
|
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
|
2017-09-05 21:20:11 -07:00
|
|
|
{:fn, :arg_names} = fn_info
|
2017-09-14 05:44:55 -07:00
|
|
|
args = {name, select(i,...) for i,name in ipairs(arg_names[fn_name])}
|
2017-08-18 17:08:15 -07:00
|
|
|
if @debug
|
2017-09-05 21:20:11 -07:00
|
|
|
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
|
2017-09-05 21:20:11 -07:00
|
|
|
fn_info = {:fn, :arg_names, :invocations, is_macro:false}
|
2017-08-22 01:02:41 -07:00
|
|
|
for invocation in *invocations
|
2017-09-05 21:20:11 -07:00
|
|
|
@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 = {}
|
2017-09-14 05:44:55 -07:00
|
|
|
arg_names = {}
|
|
|
|
prev_arg_names = nil
|
2017-08-22 01:02:41 -07:00
|
|
|
for _text in *text
|
2017-09-14 05:14:28 -07:00
|
|
|
invocation = _text\gsub("'"," '")\gsub("%%%S+","%%")\gsub("%s+"," ")
|
|
|
|
_arg_names = [arg for arg in _text\gmatch("%%(%S[^%s']*)")]
|
2017-08-22 01:02:41 -07:00
|
|
|
table.insert(invocations, invocation)
|
2017-09-14 05:44:55 -07:00
|
|
|
if prev_arg_names
|
|
|
|
if not utils.equivalent(utils.set(prev_arg_names), utils.set(_arg_names))
|
2017-09-14 05:49:01 -07:00
|
|
|
@error("Conflicting argument names #{utils.repr(prev_arg_names)} and #{utils.repr(_arg_names)} for #{utils.repr(text)}")
|
2017-09-14 05:44:55 -07:00
|
|
|
else prev_arg_names = _arg_names
|
|
|
|
arg_names[invocation] = _arg_names
|
2017-08-22 01:02:41 -07:00
|
|
|
return invocations, arg_names
|
|
|
|
|
2017-09-11 19:23:55 -07:00
|
|
|
defmacro: (spec, lua_gen_fn)=>
|
2017-09-12 15:28:58 -07:00
|
|
|
invocations,arg_names = @get_invocations spec
|
2017-09-11 19:23:55 -07:00
|
|
|
fn_info = {fn:lua_gen_fn, :arg_names, :invocations, is_macro:true}
|
2017-08-22 01:02:41 -07:00
|
|
|
for invocation in *invocations
|
2017-09-05 21:20:11 -07:00
|
|
|
@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 21:53:45 -07:00
|
|
|
-- This will execute each chunk as it goes along
|
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 21:53:45 -07:00
|
|
|
return code
|
2017-08-22 01:02:41 -07:00
|
|
|
|
|
|
|
parse: (str)=>
|
|
|
|
if @debug
|
|
|
|
print("PARSING:\n#{str}")
|
2017-09-05 21:20:11 -07:00
|
|
|
lingo = [=[
|
2017-09-14 00:09:54 -07:00
|
|
|
file <- ({ {| %blank_line* {:body: block :} %blank_line* (errors)? |} }) -> File
|
2017-09-05 22:58:27 -07:00
|
|
|
errors <- (({.+}) => error_handler)
|
2017-09-05 21:20:11 -07:00
|
|
|
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 <-
|
2017-09-14 00:09:54 -07:00
|
|
|
((".." %ws? %line_comment? (%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)?)
|
2017-09-05 21:20:11 -07:00
|
|
|
/ (fn_bit (%word_boundary fn_bits)?))
|
|
|
|
indented_fn_bits <-
|
|
|
|
fn_bit ((%new_line / %word_boundary) indented_fn_bits)?
|
|
|
|
|
|
|
|
thunk <-
|
2017-09-14 00:09:54 -07:00
|
|
|
({ ":" %ws? %line_comment?
|
2017-09-05 22:58:27 -07:00
|
|
|
((%indent %new_line block ((%dedent (%new_line "..")?) / errors))
|
2017-09-05 21:20:11 -07:00
|
|
|
/ (one_liner (%ws? (%new_line? ".."))?)) }) -> Thunk
|
|
|
|
|
2017-09-14 04:31:46 -07:00
|
|
|
word <- ({ !number {%wordchar (!"'" %wordchar)*} }) -> Word
|
2017-09-05 21:20:11 -07:00
|
|
|
expression <- ({ (longstring / string / number / variable / list / thunk / subexpression) }) -> Expression
|
|
|
|
|
2017-09-14 00:09:54 -07:00
|
|
|
string <- ({ (!longstring) '"' {(("\" [^%nl]) / [^"%nl])*} '"' }) -> String
|
2017-09-14 02:41:10 -07:00
|
|
|
longstring <- ({ '".."' %ws?
|
|
|
|
{|
|
|
|
|
(("|" {| ({("\\" / (!string_interpolation [^%nl]))+} / string_interpolation)* |})
|
|
|
|
/ %line_comment)?
|
|
|
|
(%indent
|
|
|
|
(%new_line "|" {|
|
|
|
|
({("\\" / (!string_interpolation [^%nl]))+} / string_interpolation)*
|
|
|
|
|})+
|
|
|
|
((%dedent (%new_line '..')?) / errors))?
|
|
|
|
|}}) -> Longstring
|
|
|
|
string_interpolation <- "\" %ws? (functioncall / expression) %ws? "\"
|
2017-09-05 21:20:11 -07:00
|
|
|
number <- ({ {'-'? [0-9]+ ("." [0-9]+)?} }) -> Number
|
2017-09-14 04:31:46 -07:00
|
|
|
variable <- ({ ("%" {%wordchar (!"'" %wordchar)*}) }) -> Var
|
2017-09-05 21:20:11 -07:00
|
|
|
|
|
|
|
subexpression <-
|
2017-09-14 00:09:54 -07:00
|
|
|
("(" %ws? (functioncall / expression) %ws? ")")
|
|
|
|
/ ("(..)" %ws? %line_comment? %indent %new_line ((({ {| indented_fn_bits |} }) -> FunctionCall) / expression) %dedent (%new_line "..")?)
|
2017-09-05 21:20:11 -07:00
|
|
|
|
|
|
|
list <- ({ {|
|
2017-09-14 00:09:54 -07:00
|
|
|
("[..]" %ws? %line_comment? %indent %new_line indented_list ","? ((%dedent (%new_line "..")?) / errors))
|
2017-09-05 21:20:11 -07:00
|
|
|
/ ("[" %ws? (list_items ","?)? %ws?"]")
|
|
|
|
|} }) -> List
|
2017-09-12 17:42:35 -07:00
|
|
|
list_items <- ((functioncall / expression) (list_sep list_items)?)
|
2017-09-05 21:20:11 -07:00
|
|
|
list_sep <- %ws? "," %ws?
|
|
|
|
indented_list <-
|
2017-09-14 00:09:54 -07:00
|
|
|
(functioncall / expression) (((list_sep (%line_comment? %new_line)?) / (%line_comment? %new_line)) indented_list)?
|
2017-09-05 21:20:11 -07:00
|
|
|
]=]
|
2017-09-05 23:51:35 -07:00
|
|
|
lingo = make_parser lingo
|
2017-09-05 21:20:11 -07:00
|
|
|
|
|
|
|
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
|
|
|
|
2017-09-14 02:41:10 -07:00
|
|
|
tree_to_value: (tree, vars)=>
|
|
|
|
-- TODO: clean up require utils
|
|
|
|
code = "
|
|
|
|
local utils = require('utils')
|
|
|
|
return (function(compiler, vars)\nreturn #{@tree_to_lua(tree)}\nend)"
|
2017-09-12 20:00:19 -07:00
|
|
|
lua_thunk, err = load(code)
|
2017-09-11 19:23:55 -07:00
|
|
|
if not lua_thunk
|
|
|
|
error("Failed to compile generated code:\n#{code}\n\n#{err}")
|
2017-09-14 02:41:10 -07:00
|
|
|
return (lua_thunk!)(self, vars or {})
|
2017-09-11 19:23:55 -07:00
|
|
|
|
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-14 02:41:10 -07:00
|
|
|
-- TODO: clean up repeated loading of utils?
|
|
|
|
lua_thunk, err = load("
|
|
|
|
local utils = require('utils')
|
|
|
|
return (function(compiler, vars)\n#{code}\nend)")
|
2017-09-12 20:00:19 -07:00
|
|
|
if not lua_thunk
|
2017-09-14 02:41:10 -07:00
|
|
|
error("Failed to compile generated code:\n#{code}\n\n#{err}\n\nProduced by statement:\n#{utils.repr(statement)}")
|
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)
|
2017-09-05 21:20:11 -07:00
|
|
|
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)
|
2017-09-05 21:20:11 -07:00
|
|
|
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
|
|
|
|
2017-09-05 21:20:11 -07:00
|
|
|
when "Longstring"
|
2017-09-14 02:41:10 -07:00
|
|
|
concat_parts = {}
|
|
|
|
string_buffer = ""
|
|
|
|
for i,line in ipairs(tree.value)
|
|
|
|
if i > 1 then string_buffer ..= "\n"
|
|
|
|
for bit in *line
|
|
|
|
if type(bit) == "string"
|
|
|
|
string_buffer ..= bit\gsub("\\\\","\\")
|
|
|
|
else
|
|
|
|
if string_buffer ~= ""
|
|
|
|
table.insert concat_parts, utils.repr(string_buffer, true)
|
|
|
|
string_buffer = ""
|
|
|
|
table.insert concat_parts, "utils.repr(#{to_lua(bit)})"
|
|
|
|
|
|
|
|
if string_buffer ~= ""
|
|
|
|
table.insert concat_parts, utils.repr(string_buffer, true)
|
|
|
|
|
|
|
|
if #concat_parts == 0
|
|
|
|
add "''"
|
|
|
|
elseif #concat_parts == 1
|
|
|
|
add concat_parts[1]
|
|
|
|
else
|
|
|
|
add "(#{table.concat(concat_parts, "..")})"
|
2017-09-05 21:20:11 -07:00
|
|
|
|
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"]
|
2017-09-14 05:44:55 -07:00
|
|
|
args = {name,args[i] for i,name in ipairs(arg_names[name])}
|
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"
|
2017-09-05 21:20:11 -07:00
|
|
|
-- 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)=>
|
2017-09-05 21:20:11 -07:00
|
|
|
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}")
|
2017-09-05 21:20:11 -07:00
|
|
|
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)
|
2017-09-05 21:20:11 -07:00
|
|
|
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
|
2017-09-14 02:41:10 -07:00
|
|
|
as_lua_code = (str, vars)=>
|
2017-09-12 20:00:19 -07:00
|
|
|
switch str.type
|
|
|
|
when "String"
|
2017-09-14 02:41:10 -07:00
|
|
|
return @tree_to_value(str, vars)
|
2017-09-12 20:00:19 -07:00
|
|
|
when "Longstring"
|
2017-09-14 02:41:10 -07:00
|
|
|
return @tree_to_value(str, vars)
|
2017-09-12 20:00:19 -07:00
|
|
|
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
|
2017-09-14 02:41:10 -07:00
|
|
|
return table.concat([as_lua_code(@, i.value, vars) for i in *lua_code.value]), true
|
2017-09-12 20:00:19 -07:00
|
|
|
else
|
2017-09-14 02:41:10 -07:00
|
|
|
return as_lua_code(@, lua_code, vars), true
|
2017-09-12 20:00:19 -07:00
|
|
|
|
|
|
|
@defmacro [[lua expr %lua_code]], (vars, kind)=>
|
|
|
|
lua_code = vars.lua_code.value
|
|
|
|
switch lua_code.type
|
|
|
|
when "List"
|
|
|
|
-- TODO: handle subexpressions
|
2017-09-14 02:41:10 -07:00
|
|
|
return table.concat([as_lua_code(@, i.value, vars) for i in *lua_code.value])
|
2017-09-12 20:00:19 -07:00
|
|
|
else
|
2017-09-14 02:41:10 -07:00
|
|
|
return as_lua_code(@, lua_code, vars)
|
2017-09-12 20:00:19 -07:00
|
|
|
|
|
|
|
@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.")
|
2017-09-14 02:41:10 -07:00
|
|
|
@defmacro @tree_to_value(vars.spec, vars), @tree_to_value(vars.body, vars)
|
2017-09-12 20:00:19 -07:00
|
|
|
return "", true
|
|
|
|
|
|
|
|
@defmacro [[macro block %spec %body]], (vars, kind)=>
|
|
|
|
if kind == "Expression"
|
|
|
|
error("Macro definitions cannot be used as expressions.")
|
2017-09-14 02:41:10 -07:00
|
|
|
invocation = @tree_to_value(vars.spec, vars)
|
|
|
|
fn = @tree_to_value(vars.body, vars)
|
2017-09-12 20:00:19 -07:00
|
|
|
@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'))
|
|
|
|
|
|
|
|
|
2017-09-12 22:30:41 -07:00
|
|
|
-- Run on the command line via "./nomsu.moon input_file.nom" to execute
|
|
|
|
-- and "./nomsu.moon input_file.nom output_file.lua" to compile (use "-" to compile to stdout)
|
2017-09-12 23:12:45 -07:00
|
|
|
if arg and arg[1]
|
2017-09-13 16:22:04 -07:00
|
|
|
c = NomsuCompiler()
|
2017-09-14 02:41:10 -07:00
|
|
|
--c.debug = true
|
2017-09-12 20:00:19 -07:00
|
|
|
input = io.open(arg[1])\read("*a")
|
2017-09-12 22:30:41 -07:00
|
|
|
-- Kinda hacky, if run via "./nomsu.moon file.nom -", then silence print and io.write
|
2017-09-12 20:00:19 -07:00
|
|
|
-- 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 [[
|
2017-09-14 02:41:10 -07:00
|
|
|
local utils = require('utils')
|
2017-09-12 20:00:19 -07:00
|
|
|
local load = function()
|
|
|
|
]]
|
|
|
|
output\write(code)
|
|
|
|
output\write [[
|
|
|
|
|
|
|
|
end
|
2017-09-13 16:22:04 -07:00
|
|
|
local NomsuCompiler = require('nomsu')
|
|
|
|
local c = NomsuCompiler()
|
2017-09-12 20:00:19 -07:00
|
|
|
load()(c, {})
|
|
|
|
]]
|
2017-09-14 15:35:06 -07:00
|
|
|
elseif arg
|
|
|
|
-- REPL:
|
|
|
|
c = NomsuCompiler()
|
|
|
|
while true
|
|
|
|
buff = ""
|
|
|
|
while true
|
|
|
|
io.write(">> ")
|
|
|
|
line = io.read("*L")
|
|
|
|
if line == "\n" or not line
|
|
|
|
break
|
|
|
|
buff ..= line
|
|
|
|
if #buff == 0
|
|
|
|
break
|
|
|
|
c\run(buff)
|
2017-09-12 20:00:19 -07:00
|
|
|
|
2017-09-13 16:22:04 -07:00
|
|
|
return NomsuCompiler
|