Moving tree_to_lua into each of the Nomsu tree types, which are now in
their own file.
This commit is contained in:
parent
25e06d1fce
commit
54fc7fc440
@ -13,30 +13,19 @@ immediately:
|
||||
# Conditionals
|
||||
immediately:
|
||||
compile [if %condition %if_body] to:
|
||||
%if_body <- (%if_body as lua)
|
||||
return {..}
|
||||
locals: %if_body.locals
|
||||
statements:".."
|
||||
if \(%condition as lua expr) then
|
||||
\(%if_body.statements or "\(%if_body.expr);")
|
||||
end
|
||||
Lua ".."
|
||||
if \(%condition as lua expr) then
|
||||
\(%if_body as lua statements)
|
||||
end
|
||||
parse [unless %condition %unless_body] as: if (not %condition) %unless_body
|
||||
|
||||
compile [if %condition %if_body else %else_body, unless %condition %else_body else %if_body] to:
|
||||
%if_body <- (%if_body as lua)
|
||||
%else_body <- (%else_body as lua)
|
||||
lua> ".."
|
||||
local \%locals = {unpack(\%if_body.locals or {})};
|
||||
for i,loc in ipairs(\%else_body.locals or {}) do table.insert(\%locals, loc); end
|
||||
utils.deduplicate(\%locals);
|
||||
return {..}
|
||||
locals:%locals
|
||||
statements:".."
|
||||
if \(%condition as lua expr) then
|
||||
\(%if_body.statements or "\(%if_body.expr);")
|
||||
else
|
||||
\(%else_body.statements or "\(%else_body.expr);")
|
||||
end
|
||||
Lua ".."
|
||||
if \(%condition as lua expr) then
|
||||
\(%if_body as lua statements)
|
||||
else
|
||||
\(%else_body as lua statements)
|
||||
end
|
||||
|
||||
# Conditional expression (ternary operator)
|
||||
#.. Note: this uses a function instead of "(condition and if_expr or else_expr)"
|
||||
@ -51,14 +40,15 @@ immediately
|
||||
#.. If %when_true_expr is guaranteed to be truthy, we can use Lua's idiomatic
|
||||
equivalent of a conditional expression: (cond and if_true or if_false)
|
||||
if: %when_true_expr.type in {Text:yes, List:yes, Dict:yes, Number:yes}
|
||||
return {..}
|
||||
expr:"(\(%condition as lua expr) and \(%when_true_expr as lua expr) or \(%when_false_expr as lua expr))"
|
||||
return:
|
||||
Lua ".."
|
||||
(\(%condition as lua expr) and \(%when_true_expr as lua expr) or \(%when_false_expr as lua expr))
|
||||
..else:
|
||||
#.. Otherwise, need to do an anonymous inline function (yuck, too bad lua
|
||||
doesn't have a proper ternary operator!)
|
||||
To see why this is necessary consider: (random()<.5 and false or 99)
|
||||
return {..}
|
||||
expr: ".."
|
||||
return:
|
||||
Lua ".."
|
||||
(function()
|
||||
if \(%condition as lua expr) then
|
||||
return \(%when_true_expr as lua expr);
|
||||
@ -82,67 +72,59 @@ immediately:
|
||||
# Helper function
|
||||
immediately:
|
||||
compile [if %tree has subtree %subtree where %condition %body] to:
|
||||
%body_lua <- (%body as lua)
|
||||
%body_statements <- (%body_lua.statements or "\(%body_lua.expr);")
|
||||
return {..}
|
||||
locals: %body_lua.locals
|
||||
statements:".."
|
||||
for \(%subtree as lua expr) in coroutine.wrap(function() nomsu:walk_tree(\(%tree as lua expr)) end) do
|
||||
if Types.is_node(\(%subtree as lua expr)) then
|
||||
if \(%condition as lua expr) then
|
||||
\%body_statements
|
||||
break;
|
||||
end
|
||||
Lua ".."
|
||||
for \(%subtree as lua expr) in coroutine.wrap(function() nomsu:walk_tree(\(%tree as lua expr)) end) do
|
||||
if Types.is_node(\(%subtree as lua expr)) then
|
||||
if \(%condition as lua expr) then
|
||||
\(%body as lua statements)
|
||||
break;
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
# While loops
|
||||
immediately:
|
||||
compile [do next repeat] to {statements:"goto continue_repeat;"}
|
||||
compile [stop repeating] to {statements:"goto stop_repeat;"}
|
||||
compile [repeat while %condition %body] to
|
||||
%body_lua <- (%body as lua)
|
||||
%body_statements <- (%body_lua.statements or "\(%body_lua.expr);")
|
||||
%lua <-: Lua "while \(%condition as lua expr) do"
|
||||
if %body has subtree % where:
|
||||
(%.type = "FunctionCall") and ((%'s stub) is "do next repeat")
|
||||
..: %body_statments +<- "\n::continue_repeat::;"
|
||||
%code <- ".."
|
||||
while \(%condition as lua expr) do
|
||||
\%body_statements
|
||||
end --while-loop
|
||||
..:
|
||||
%lua +<- "\n::continue_repeat::;"
|
||||
%lua +<- "\n"
|
||||
%lua +<- (%body as lua statements)
|
||||
%lua +<- "\nend --while-loop"
|
||||
if %body has subtree % where:
|
||||
(%.type = "FunctionCall") and ((%'s stub) is "stop repeating")
|
||||
..:
|
||||
%code <- ".."
|
||||
%lua <- ".."
|
||||
do -- scope of "stop repeating" label
|
||||
\%code
|
||||
\%lua
|
||||
::stop_repeat::;
|
||||
end -- end of "stop repeating" label scope
|
||||
return {statements:%code, locals:%body_lua.locals}
|
||||
return %lua
|
||||
parse [repeat %body] as: repeat while (yes) %body
|
||||
parse [repeat until %condition %body] as: repeat while (not %condition) %body
|
||||
|
||||
compile [..]
|
||||
repeat %n times %body
|
||||
..to:
|
||||
%body_lua <- (%body as lua)
|
||||
%body_statements <- (%body_lua.statements or "\(%body_lua.expr);")
|
||||
%lua <-: Lua "for i=1,\(%n as lua expr) do"
|
||||
if %body has subtree % where
|
||||
(%.type = "FunctionCall") and ((%'s stub) is "do next repeat")
|
||||
..: %body_statements +<- "\n::continue_repeat::;"
|
||||
%code <- ".."
|
||||
for i=1,\(%n as lua expr) do
|
||||
\%body_statements
|
||||
end --numeric for-loop
|
||||
..: %lua +<- "\n::continue_repeat::;"
|
||||
%lua +<- "\n\(%body as lua statements)\nend --numeric for-loop"
|
||||
if %body has subtree % where:
|
||||
(%.type = "FunctionCall") and ((%'s stub) is "stop repeating")
|
||||
..:
|
||||
%code <- ".."
|
||||
do -- scope of "stop repeating" label
|
||||
\%code
|
||||
::stop_repeat::;
|
||||
end -- end of "stop repeating" label scope
|
||||
return {statements:%code, locals:%body_lua.locals}
|
||||
%lua <-:
|
||||
Lua ".."
|
||||
do -- scope of "stop repeating" label
|
||||
\%lua
|
||||
::stop_repeat::;
|
||||
end -- end of "stop repeating" label scope
|
||||
return %lua
|
||||
|
||||
# For loop control flow:
|
||||
immediately:
|
||||
@ -157,33 +139,30 @@ immediately:
|
||||
for %var from %start to %stop by %step %body
|
||||
for %var from %start to %stop via %step %body
|
||||
..to:
|
||||
%body_lua <- (%body as lua)
|
||||
%body_statements <- (%body_lua.statements or "\(%body_lua.expr);")
|
||||
# This uses Lua's approach of only allowing loop-scoped variables in a loop
|
||||
assume (%var.type is "Var") or barf "Loop expected variable, not: \(%var's source code)"
|
||||
%lua <-
|
||||
Lua ".."
|
||||
for \(%var as lua expr)=\(%start as lua expr),\(%n as lua expr) do
|
||||
if %body has subtree % where:
|
||||
(%.type = "FunctionCall") and
|
||||
((%'s stub) is "do next %") and
|
||||
%.value.3.value is %var.value
|
||||
..: %body_statements +<- "\n::continue_\(%var as lua identifier)::;"
|
||||
|
||||
# This uses Lua's approach of only allowing loop-scoped variables in a loop
|
||||
assume (%var.type is "Var") or barf "Loop expected variable, not: \(%var's source code)"
|
||||
%code <- ".."
|
||||
for \(%var as lua expr)=\(%start as lua expr),\(%stop as lua expr),\(%step as lua expr) do
|
||||
\%body_statements
|
||||
end --numeric for-loop
|
||||
..: %lua write code "\n::continue_\(%var as lua identifier)::;"
|
||||
%lua write code "\n\(%body as lua statements)\nend --numeric for-loop"
|
||||
|
||||
if %body has subtree % where:
|
||||
(%.type = "FunctionCall") and:
|
||||
((%'s stub) is "stop %") and:
|
||||
%.value.2.value is %var.value
|
||||
..:
|
||||
%code <- ".."
|
||||
%lua write code ".."
|
||||
do -- scope for stopping for-loop
|
||||
\%code
|
||||
\%lua
|
||||
::stop_\(%var as lua identifier)::;
|
||||
end -- end of scope for stopping for-loop
|
||||
|
||||
return {statements:%code, locals:%body_lua.locals}
|
||||
return %lua
|
||||
|
||||
immediately:
|
||||
parse [for %var from %start to %stop %body] as: for %var from %start to %stop via 1 %body
|
||||
@ -196,57 +175,49 @@ immediately:
|
||||
# For-each loop (lua's "ipairs()")
|
||||
immediately:
|
||||
compile [for %var in %iterable %body] to:
|
||||
%body_lua <- (%body as lua)
|
||||
%body_statements <- (%body_lua.statements or "\(%body_lua.expr);")
|
||||
# This uses Lua's approach of only allowing loop-scoped variables in a loop
|
||||
assume (%var.type is "Var") or barf "Loop expected variable, not: \(%var's source code)"
|
||||
%lua <-: Lua "for i,\(%var as lua identifier) in ipairs(\(%iterable as lua expr)) do"
|
||||
if %body has subtree % where:
|
||||
(%.type = "FunctionCall") and
|
||||
((%'s stub) is "do next %") and
|
||||
%.value.3.value is %var.value
|
||||
..: %body_statements +<- "\n::continue_\(%var as lua identifier)::;"
|
||||
# This uses Lua's approach of only allowing loop-scoped variables in a loop
|
||||
assume (%var.type is "Var") or barf "Loop expected variable, not: \(%var's source code)"
|
||||
%code <- ".."
|
||||
for i,\(%var as lua expr) in ipairs(\(%iterable as lua expr)) do
|
||||
\%body_statements
|
||||
end --foreach-loop
|
||||
..: %lua +<- (Lua "\n::continue_\(%var as lua identifier)::;")
|
||||
%lua +<- (Lua "\n\(%body as lua statements)\nend --foreach-loop")
|
||||
if %body has subtree % where:
|
||||
(%.type = "FunctionCall") and
|
||||
((%'s stub) is "stop %") and
|
||||
%.value.2.value is %var.value
|
||||
..:
|
||||
%code <- ".."
|
||||
do -- scope for stopping for-loop
|
||||
\%code
|
||||
::stop_\(%var as lua identifier)::;
|
||||
end -- end of scope for stopping for-loop
|
||||
return {statements:%code, locals:%body_lua.locals}
|
||||
%lua <-
|
||||
Lua ".."
|
||||
do -- scope for stopping for-loop
|
||||
\%lua
|
||||
::stop_\(%var as lua identifier)::;
|
||||
end -- end of scope for stopping for-loop
|
||||
return %lua
|
||||
|
||||
parse [for all %iterable %body] as: for % in %iterable %body
|
||||
|
||||
# Dict iteration (lua's "pairs()")
|
||||
immediately:
|
||||
compile [for %key = %value in %iterable %body] to:
|
||||
%body_lua <- (%body as lua)
|
||||
%body_statements <- (%body_lua.statements or "\(%body_lua.expr);")
|
||||
# This uses Lua's approach of only allowing loop-scoped variables in a loop
|
||||
assume (%key.type is "Var") or barf "Loop expected variable, not: \(%key's source code)"
|
||||
assume (%value.type is "Var") or barf "Loop expected variable, not: \(%value's source code)"
|
||||
%lua <- (Lua "for \(%key as lua identifier),\(%value as lua identifier) in pairs(\(%iterable as lua expr)) do")
|
||||
if %body has subtree % where:
|
||||
(%.type = "FunctionCall") and
|
||||
((%'s stub) is "do next %") and
|
||||
%.value.3.value is %key.value
|
||||
..: %body_statements +<- "\n::continue_\(%key as lua identifier)::;"
|
||||
..: %lua +<- (Lua "\n::continue_\(%key as lua identifier)::;")
|
||||
|
||||
if %body has subtree % where:
|
||||
(%.type = "FunctionCall") and
|
||||
((%'s stub) is "do next %") and
|
||||
%.value.3.value is %value.value
|
||||
..: %body_statements +<- "\n::continue_\(%value as lua identifier)::;"
|
||||
|
||||
# This uses Lua's approach of only allowing loop-scoped variables in a loop
|
||||
assume (%key.type is "Var") or barf "Loop expected variable, not: \(%key's source code)"
|
||||
assume (%value.type is "Var") or barf "Loop expected variable, not: \(%value's source code)"
|
||||
%code <- ".."
|
||||
for \(%key as lua expr),\(%value as lua expr) in pairs(\(%iterable as lua expr)) do
|
||||
\%body_statements
|
||||
end --foreach-loop
|
||||
..: %lua +<- (Lua "\n::continue_\(%value as lua identifier)::;")
|
||||
%lua +<- (Lua "\n\(%body as lua statements)\nend --foreach-loop")
|
||||
|
||||
%stop_labels <- ""
|
||||
if %body has subtree % where:
|
||||
@ -262,11 +233,12 @@ immediately:
|
||||
..: %stop_labels +<- "\n::stop_\(%value as lua identifier)::;"
|
||||
|
||||
if: %stop_labels is not ""
|
||||
%code <- ".."
|
||||
do -- scope for stopping for % = % loop
|
||||
\%code\%stop_labels
|
||||
end
|
||||
return {statements:%code, locals:%body_lua.locals}
|
||||
%lua <-
|
||||
Lua ".."
|
||||
do -- scope for stopping for % = % loop
|
||||
\%lua\%stop_labels
|
||||
end
|
||||
return %lua
|
||||
|
||||
# Switch statement/multi-branch if
|
||||
immediately:
|
||||
@ -279,11 +251,11 @@ immediately:
|
||||
for %func_call in %body.value:
|
||||
assume (%func_call.type is "FunctionCall") or barf ".."
|
||||
Invalid format for 'when' statement. Only '*' blocks are allowed.
|
||||
with [..]
|
||||
%tokens <- %func_call.value
|
||||
%star <- (1st in %tokens)
|
||||
%condition <- (2nd in %tokens)
|
||||
%action <- (3rd in %tokens)
|
||||
with {..}
|
||||
%tokens: %func_call.value
|
||||
%star: %tokens.1
|
||||
%condition: %tokens.2
|
||||
%action: %tokens.3
|
||||
..:
|
||||
assume (=lua "\%star and \%star.type == 'Word' and \%star.value == '*'") or barf ".."
|
||||
Invalid format for 'when' statement. Lines must begin with '*'
|
||||
@ -334,7 +306,7 @@ immediately:
|
||||
assume (%func_call.type is "FunctionCall") or barf ".."
|
||||
Invalid format for 'when' statement. Only '*' blocks are allowed.
|
||||
%tokens <- %func_call.value
|
||||
with [%star<-(1st in %tokens), %condition<-(2nd in %tokens), %action<-(3rd in %tokens)]:
|
||||
with {%star:%tokens.1, %condition:%tokens.2, %action:%tokens.3}:
|
||||
assume (=lua "\%star and \%star.type == 'Word' and \%star.value == '*'") or barf ".."
|
||||
Invalid format for 'when' statement. Lines must begin with '*'
|
||||
assume %condition or barf ".."
|
||||
|
@ -9,7 +9,7 @@ immediately:
|
||||
local lua = Lua(tree.source, "nomsu:define_compile_action(");
|
||||
local stubs = {};
|
||||
for i, action in ipairs(\%actions.value) do
|
||||
stubs[i] = nomsu:tree_to_named_stub(action);
|
||||
stubs[i] = action:get_stub(true);
|
||||
end
|
||||
lua:append(repr(stubs), ", ", repr(tree.source:get_line()), ", function(tree");
|
||||
local args = {};
|
||||
@ -23,7 +23,8 @@ immediately:
|
||||
lua:append(", ");
|
||||
lua:append(arg);
|
||||
end
|
||||
local body_lua = nomsu:tree_to_lua(\%lua):as_statements("return ");
|
||||
local body_lua = \%lua:as_lua(nomsu);
|
||||
body_lua:convert_to_statements("return ");
|
||||
body_lua:declare_locals(args);
|
||||
lua:append(")\\n ", body_lua, "\\nend);")
|
||||
return lua;
|
||||
@ -36,7 +37,7 @@ immediately:
|
||||
local lua = Lua(tree.source, "nomsu:define_action(");
|
||||
local stubs = {};
|
||||
for i, action in ipairs(\%actions.value) do
|
||||
stubs[i] = nomsu:tree_to_named_stub(action);
|
||||
stubs[i] = action:get_stub(true);
|
||||
end
|
||||
lua:append(repr(stubs), ", ", repr(tree.source:get_line()), ", function(");
|
||||
local args = {};
|
||||
@ -47,7 +48,8 @@ immediately:
|
||||
lua:append(arg);
|
||||
if i < #args then lua:append(", ") end
|
||||
end
|
||||
local body_lua = nomsu:tree_to_lua(\%body):as_statements("return ");
|
||||
local body_lua = \%body:as_lua(nomsu);
|
||||
body_lua:convert_to_statements("return ");
|
||||
body_lua:declare_locals(args);
|
||||
lua:append(")\\n ", body_lua, "\\nend);")
|
||||
return lua;
|
||||
@ -59,7 +61,7 @@ immediately:
|
||||
local lua = Lua(tree.source, "nomsu:define_compile_action(");
|
||||
local stubs = {};
|
||||
for i, action in ipairs(\%shorthand.value) do
|
||||
stubs[i] = nomsu:tree_to_named_stub(action);
|
||||
stubs[i] = action:get_stub(true);
|
||||
end
|
||||
lua:append(repr(stubs), ", ", repr(tree.source:get_line()), ", function(tree");
|
||||
local args = {};
|
||||
@ -84,7 +86,7 @@ immediately:
|
||||
lua:append([[)
|
||||
local template = nomsu:parse(]]..template..[[, ]]..repr(tree.source.filename)..[[);
|
||||
local replacement = nomsu:tree_with_replaced_vars(template, ]]..replacements..[[);
|
||||
return nomsu:tree_to_lua(replacement, nomsu.compilestack[#nomsu.compilestack].source.filename);
|
||||
return replacement:as_lua(nomsu);
|
||||
end);
|
||||
]]);
|
||||
return lua;
|
||||
@ -101,11 +103,11 @@ action [remove action %stub]:
|
||||
|
||||
immediately:
|
||||
action [%tree as lua]:
|
||||
=lua "nomsu:tree_to_lua(\%tree)"
|
||||
=lua "\%tree:as_lua(nomsu)"
|
||||
|
||||
action [%tree as lua expr]:
|
||||
lua> ".."
|
||||
local lua = nomsu:tree_to_lua(\%tree);
|
||||
local lua = \%tree:as_lua(nomsu);
|
||||
if not lua.is_value then
|
||||
error("Invalid thing to convert to lua expr: "..\%tree.source:get_text());
|
||||
end
|
||||
@ -113,7 +115,8 @@ immediately:
|
||||
|
||||
action [%tree as lua statements]:
|
||||
lua> ".."
|
||||
local lua = nomsu:tree_to_lua(\%tree):as_statements();
|
||||
local lua = \%tree:as_lua(nomsu);
|
||||
lua:convert_to_statements();
|
||||
lua:declare_locals();
|
||||
return lua;
|
||||
|
||||
@ -124,14 +127,17 @@ immediately:
|
||||
=lua "nomsu:tree_to_stub(\%tree)"
|
||||
|
||||
immediately:
|
||||
compile [%tree's source code, %tree' source code] to: LuaValue "\(%tree as lua expr).source:get_text()"
|
||||
|
||||
compile [repr %obj] to: LuaValue "repr(\(%obj as lua expr))"
|
||||
compile [type of %obj] to: LuaValue "type(\(%obj as lua expr))"
|
||||
parse [%var write code %code] as: lua> "\%var:append(\%code);"
|
||||
|
||||
immediately:
|
||||
compile [nomsu] to: LuaValue "nomsu"
|
||||
compile [%var as lua identifier] to: LuaValue "nomsu:var_to_lua_identifier(\(%var as lua expr))"
|
||||
compile [%tree's source code, %tree' source code] to: Lua value "\(%tree as lua expr).source:get_text()"
|
||||
|
||||
compile [repr %obj] to: Lua value "repr(\(%obj as lua expr))"
|
||||
compile [type of %obj] to: Lua value "type(\(%obj as lua expr))"
|
||||
|
||||
immediately:
|
||||
compile [nomsu] to: Lua value "nomsu"
|
||||
compile [%var as lua identifier] to: Lua value "nomsu:var_to_lua_identifier(\(%var as lua expr))"
|
||||
|
||||
action [action %names metadata]:
|
||||
=lua "nomsu.action_metadata[ACTIONS[\%names]]"
|
||||
@ -149,7 +155,7 @@ action [help %action]:
|
||||
# Compiler tools
|
||||
immediately:
|
||||
compile [run %code] to:
|
||||
LuaValue ".."
|
||||
Lua value ".."
|
||||
nomsu:run(\(%code as lua expr), '\
|
||||
=lua "nomsu:get_line_number(nomsu.compilestack[#nomsu.compilestack])"
|
||||
..')
|
||||
@ -159,7 +165,7 @@ immediately:
|
||||
immediately:
|
||||
compile [show lua %block] to:
|
||||
lua> ".."
|
||||
local \%lua = nomsu:tree_to_lua(\%block);
|
||||
local \%lua = \%block:as_lua(nomsu);
|
||||
return Lua(\%block.source, "print(", repr(tostring(\%lua)), ");");
|
||||
|
||||
immediately:
|
||||
@ -172,6 +178,13 @@ immediately:
|
||||
return Lua(tree.source, "print(stringify(", \(%message as lua expr), "));");
|
||||
end
|
||||
|
||||
immediately:
|
||||
compile [source] to: Lua value (=lua "tree.source") "tree.source"
|
||||
|
||||
immediately:
|
||||
action [Lua %]: Lua (=lua "nomsu.compilestack[#nomsu.compilestack]") %
|
||||
action [Lua value %]: Lua value (=lua "nomsu.compilestack[#nomsu.compilestack]") %
|
||||
|
||||
# Return
|
||||
immediately:
|
||||
#.. Return statement is wrapped in a do..end block because Lua is unhappy if you
|
||||
@ -181,8 +194,8 @@ immediately:
|
||||
|
||||
# Error functions
|
||||
immediately:
|
||||
compile [barf] to: LuaValue "error(nil, 0);"
|
||||
compile [barf %msg] to: LuaValue "error(\(%msg as lua expr), 0);"
|
||||
compile [barf] to: Lua value "error(nil, 0);"
|
||||
compile [barf %msg] to: Lua value "error(\(%msg as lua expr), 0);"
|
||||
compile [assume %condition] to:
|
||||
lua> "local \%assumption = 'Assumption failed: '..\%condition.source:get_text();"
|
||||
return:
|
||||
@ -200,7 +213,7 @@ immediately:
|
||||
|
||||
# Literals
|
||||
immediately:
|
||||
compile [yes] to: LuaValue "true"
|
||||
compile [no] to: LuaValue "false"
|
||||
compile [nothing, nil, null] to: LuaValue "nil"
|
||||
compile [yes] to: Lua value "true"
|
||||
compile [no] to: Lua value "false"
|
||||
compile [nothing, nil, null] to: Lua value "nil"
|
||||
|
||||
|
@ -18,31 +18,31 @@ immediately:
|
||||
|
||||
# Comparison Operators
|
||||
immediately:
|
||||
compile [%x < %y] to: LuaValue "(\(%x as lua expr) < \(%y as lua expr))"
|
||||
compile [%x > %y] to: LuaValue "(\(%x as lua expr) > \(%y as lua expr))"
|
||||
compile [%x <= %y] to: LuaValue "(\(%x as lua expr) <= \(%y as lua expr))"
|
||||
compile [%x >= %y] to: LuaValue "(\(%x as lua expr) >= \(%y as lua expr))"
|
||||
compile [%x < %y] to: Lua value "(\(%x as lua expr) < \(%y as lua expr))"
|
||||
compile [%x > %y] to: Lua value "(\(%x as lua expr) > \(%y as lua expr))"
|
||||
compile [%x <= %y] to: Lua value "(\(%x as lua expr) <= \(%y as lua expr))"
|
||||
compile [%x >= %y] to: Lua value "(\(%x as lua expr) >= \(%y as lua expr))"
|
||||
# TODO: optimize case of [%x,%y] = [1,2]
|
||||
compile [%a is %b, %a = %b, %a == %b] to:
|
||||
lua> ".."
|
||||
local safe = {Text=true, Number=true};
|
||||
local a_lua, b_lua = nomsu:tree_to_lua(\%a), nomsu:tree_to_lua(\%b);
|
||||
if safe[\%a.type] or safe[\%b.type] then
|
||||
return LuaValue(tree.source, "(", a_lua, " == ", b_lua, ")");
|
||||
return Lua.Value(tree.source, "(", a_lua, " == ", b_lua, ")");
|
||||
else
|
||||
return LuaValue(tree.source, "utils.equivalent(", a_lua, ", ", b_lua, ")");
|
||||
return Lua.Value(tree.source, "utils.equivalent(", a_lua, ", ", b_lua, ")");
|
||||
end
|
||||
compile [%a isn't %b, %a is not %b, %a not= %b, %a != %b] to:
|
||||
lua> ".."
|
||||
local safe = {Text=true, Number=true};
|
||||
local a_lua, b_lua = nomsu:tree_to_lua(\%a), nomsu:tree_to_lua(\%b);
|
||||
if safe[\%a.type] or safe[\%b.type] then
|
||||
return LuaValue(tree.source, "(", a_lua, " ~= ", b_lua, ")");
|
||||
return Lua.Value(tree.source, "(", a_lua, " ~= ", b_lua, ")");
|
||||
else
|
||||
return LuaValue(tree.source, "(not utils.equivalent(", a_lua, ", ", b_lua, "))");
|
||||
return Lua.Value(tree.source, "(not utils.equivalent(", a_lua, ", ", b_lua, "))");
|
||||
end
|
||||
# For strict identity checking, use (%x's id) is (%y's id)
|
||||
compile [%'s id, id of %] to: LuaValue "nomsu.ids[\(% as lua expr)]"
|
||||
compile [%'s id, id of %] to: Lua value "nomsu.ids[\(% as lua expr)]"
|
||||
|
||||
# Variable assignment operator
|
||||
immediately:
|
||||
@ -72,7 +72,7 @@ immediately:
|
||||
local value_lua = nomsu:tree_to_lua(value);
|
||||
if not value_lua.is_value then error("Invalid value for assignment: "..value:get_src()); end
|
||||
if target.type == "Var" then
|
||||
lhs:add_free_var(nomsu:var_to_lua_identifier(target.value));
|
||||
lhs:add_free_vars(nomsu:var_to_lua_identifier(target.value));
|
||||
end
|
||||
if i > 1 then
|
||||
lhs:append(", ");
|
||||
@ -104,12 +104,12 @@ immediately:
|
||||
|
||||
immediately:
|
||||
# Math Operators
|
||||
compile [%x + %y] to: LuaValue "(\(%x as lua expr) + \(%y as lua expr))"
|
||||
compile [%x - %y] to: LuaValue "(\(%x as lua expr) - \(%y as lua expr))"
|
||||
compile [%x * %y] to: LuaValue "(\(%x as lua expr) * \(%y as lua expr))"
|
||||
compile [%x / %y] to: LuaValue "(\(%x as lua expr) / \(%y as lua expr))"
|
||||
compile [%x ^ %y] to: LuaValue "(\(%x as lua expr) ^ \(%y as lua expr))"
|
||||
compile [%x wrapped around %y, %x mod %y] to: LuaValue "(\(%x as lua expr) % \(%y as lua expr))"
|
||||
compile [%x + %y] to: Lua value "(\(%x as lua expr) + \(%y as lua expr))"
|
||||
compile [%x - %y] to: Lua value "(\(%x as lua expr) - \(%y as lua expr))"
|
||||
compile [%x * %y] to: Lua value "(\(%x as lua expr) * \(%y as lua expr))"
|
||||
compile [%x / %y] to: Lua value "(\(%x as lua expr) / \(%y as lua expr))"
|
||||
compile [%x ^ %y] to: Lua value "(\(%x as lua expr) ^ \(%y as lua expr))"
|
||||
compile [%x wrapped around %y, %x mod %y] to: Lua value "(\(%x as lua expr) % \(%y as lua expr))"
|
||||
|
||||
# 3-part chained comparisons
|
||||
# (uses a lambda to avoid re-evaluating middle value, while still being an expression)
|
||||
@ -124,22 +124,22 @@ immediately:
|
||||
# TODO: optimize for common case where x,y,z are all either variables or number literals
|
||||
|
||||
# Boolean Operators
|
||||
compile [%x and %y] to: LuaValue "(\(%x as lua expr) and \(%y as lua expr))"
|
||||
compile [%x or %y] to: LuaValue "(\(%x as lua expr) or \(%y as lua expr))"
|
||||
compile [%x and %y] to: Lua value "(\(%x as lua expr) and \(%y as lua expr))"
|
||||
compile [%x or %y] to: Lua value "(\(%x as lua expr) or \(%y as lua expr))"
|
||||
|
||||
# Bitwise Operators
|
||||
compile [%a OR %b, %a | %b] to: LuaValue "bit32.bor(\(%a as lua expr), \(%b as lua expr))"
|
||||
compile [%a XOR %b] to: LuaValue "bit32.bxor(\(%a as lua expr), \(%b as lua expr))"
|
||||
compile [%a AND %b, %a & %b] to: LuaValue "bit32.band(\(%a as lua expr), \(%b as lua expr))"
|
||||
compile [NOT %, ~ %] to: LuaValue "bit32.bnot(\(% as lua expr))"
|
||||
compile [%x LSHIFT %shift, %x << %shift] to: LuaValue "bit32.lshift(\(%x as lua expr), \(%shift as lua expr))"
|
||||
compile [%x RSHIFT %shift, %x >>> %shift] to: LuaValue "bit32.rshift(\(%x as lua expr), \(%shift as lua expr))"
|
||||
compile [%x ARSHIFT %shift, %x >> %shift] to: LuaValue "bit32.arshift(\(%x as lua expr), \(%shift as lua expr))"
|
||||
compile [%a OR %b, %a | %b] to: Lua value "bit32.bor(\(%a as lua expr), \(%b as lua expr))"
|
||||
compile [%a XOR %b] to: Lua value "bit32.bxor(\(%a as lua expr), \(%b as lua expr))"
|
||||
compile [%a AND %b, %a & %b] to: Lua value "bit32.band(\(%a as lua expr), \(%b as lua expr))"
|
||||
compile [NOT %, ~ %] to: Lua value "bit32.bnot(\(% as lua expr))"
|
||||
compile [%x LSHIFT %shift, %x << %shift] to: Lua value "bit32.lshift(\(%x as lua expr), \(%shift as lua expr))"
|
||||
compile [%x RSHIFT %shift, %x >>> %shift] to: Lua value "bit32.rshift(\(%x as lua expr), \(%shift as lua expr))"
|
||||
compile [%x ARSHIFT %shift, %x >> %shift] to: Lua value "bit32.arshift(\(%x as lua expr), \(%shift as lua expr))"
|
||||
# TODO: implement OR, XOR, AND for multiple operands?
|
||||
|
||||
# Unary operators
|
||||
compile [- %] to: LuaValue "(- \(% as lua expr))"
|
||||
compile [not %] to: LuaValue "(not \(% as lua expr))"
|
||||
compile [- %] to: Lua value "(- \(% as lua expr))"
|
||||
compile [not %] to: Lua value "(not \(% as lua expr))"
|
||||
|
||||
# Update operators
|
||||
immediately:
|
||||
|
153
lua_obj.lua
153
lua_obj.lua
@ -4,7 +4,7 @@ do
|
||||
insert, remove, concat = _obj_0.insert, _obj_0.remove, _obj_0.concat
|
||||
end
|
||||
local immutable = require('immutable')
|
||||
local Lua, LuaValue, Location
|
||||
local Lua, Location
|
||||
Location = immutable({
|
||||
"filename",
|
||||
"start",
|
||||
@ -12,7 +12,6 @@ Location = immutable({
|
||||
}, {
|
||||
name = "Location",
|
||||
__new = function(self, filename, start, stop)
|
||||
assert(type(filename) == 'string' and type(start) == 'number' and type(stop) == 'number')
|
||||
return filename, start, stop or start
|
||||
end,
|
||||
__tostring = function(self)
|
||||
@ -38,13 +37,14 @@ Location = immutable({
|
||||
return FILE_CACHE[self.filename]:sub(self.start, self.stop)
|
||||
end,
|
||||
get_line_number = function(self)
|
||||
local line_starts = LINE_STARTS[FILE_CACHE[self.filename]]
|
||||
local src = FILE_CACHE[self.filename]
|
||||
local line_starts = LINE_STARTS[src]
|
||||
local start_line = 1
|
||||
while (line_starts[start_line + 1] or (#src + 1)) <= self.start do
|
||||
while (line_starts[start_line + 1] or math.huge) <= self.start do
|
||||
start_line = start_line + 1
|
||||
end
|
||||
local stop_line = start_line
|
||||
while (line_starts[stop_line + 1] or (#src + 1)) <= self.stop do
|
||||
while (line_starts[stop_line + 1] or math.huge) <= self.stop do
|
||||
stop_line = stop_line + 1
|
||||
end
|
||||
return start_line, stop_line
|
||||
@ -64,9 +64,17 @@ Location = immutable({
|
||||
do
|
||||
local _class_0
|
||||
local _base_0 = {
|
||||
is_statement = true,
|
||||
is_value = false,
|
||||
add_free_vars = function(self, free_vars)
|
||||
clone = function(self)
|
||||
local copy = Lua(self.source, {
|
||||
unpack(self.bits)
|
||||
})
|
||||
copy.is_value = self.is_value
|
||||
for k, v in pairs(self.free_vars) do
|
||||
copy.free_vars[k] = v
|
||||
end
|
||||
return copy
|
||||
end,
|
||||
add_free_vars = function(self, ...)
|
||||
local seen
|
||||
do
|
||||
local _tbl_0 = { }
|
||||
@ -80,16 +88,30 @@ do
|
||||
end
|
||||
seen = _tbl_0
|
||||
end
|
||||
for _index_0 = 1, #free_vars do
|
||||
local var = free_vars[_index_0]
|
||||
for i = 1, select("#", ...) do
|
||||
local var = select(i, ...)
|
||||
if not (seen[var]) then
|
||||
self.free_vars[#self.free_vars + 1] = var
|
||||
seen[var] = true
|
||||
end
|
||||
end
|
||||
end,
|
||||
as_statements = function(self)
|
||||
return self
|
||||
convert_to_statements = function(self, prefix, suffix)
|
||||
if prefix == nil then
|
||||
prefix = ""
|
||||
end
|
||||
if suffix == nil then
|
||||
suffix = ";"
|
||||
end
|
||||
if not (self.is_value) then
|
||||
return
|
||||
end
|
||||
if prefix ~= "" then
|
||||
self:prepend(prefix)
|
||||
end
|
||||
if suffix ~= "" then
|
||||
return self:append(suffix)
|
||||
end
|
||||
end,
|
||||
declare_locals = function(self, skip)
|
||||
if skip == nil then
|
||||
@ -128,12 +150,11 @@ do
|
||||
local buff = { }
|
||||
for i, b in ipairs(self.bits) do
|
||||
buff[#buff + 1] = tostring(b)
|
||||
if i < #self.bits and type(b) ~= 'string' and b.is_statement then
|
||||
if i < #self.bits and type(b) ~= 'string' and not b.is_value then
|
||||
buff[#buff + 1] = "\n"
|
||||
end
|
||||
end
|
||||
local ret = concat(buff, "")
|
||||
assert(not ret:match(".*table: 0x.*"))
|
||||
return ret
|
||||
end,
|
||||
__len = function(self)
|
||||
@ -145,12 +166,16 @@ do
|
||||
end
|
||||
return len
|
||||
end,
|
||||
__add = function(self, other)
|
||||
return Lua(nil, self, other)
|
||||
end,
|
||||
__concat = function(self, other)
|
||||
return Lua(nil, self, other)
|
||||
end,
|
||||
append = function(self, ...)
|
||||
local n = select("#", ...)
|
||||
local bits = self.bits
|
||||
for i = 1, n do
|
||||
local x = select(i, ...)
|
||||
assert(type(x) ~= 'table' or getmetatable(x))
|
||||
bits[#bits + 1] = select(i, ...)
|
||||
end
|
||||
end,
|
||||
@ -158,13 +183,9 @@ do
|
||||
local n = select("#", ...)
|
||||
local bits = self.bits
|
||||
for i = #bits + n, n + 1, -1 do
|
||||
local x = select(i, ...)
|
||||
assert(type(x) ~= 'table' or getmetatable(x))
|
||||
bits[i] = bits[i - n]
|
||||
end
|
||||
for i = 1, n do
|
||||
local x = select(i, ...)
|
||||
assert(type(x) ~= 'table' or getmetatable(x))
|
||||
bits[i] = select(i, ...)
|
||||
end
|
||||
end,
|
||||
@ -197,6 +218,14 @@ do
|
||||
end
|
||||
walk(self)
|
||||
return lua_str, metadata
|
||||
end,
|
||||
parenthesize = function(self)
|
||||
if self.is_value then
|
||||
self:prepend("(")
|
||||
return self:append(")")
|
||||
else
|
||||
return error("Cannot parenthesize lua statements")
|
||||
end
|
||||
end
|
||||
}
|
||||
_base_0.__index = _base_0
|
||||
@ -207,14 +236,11 @@ do
|
||||
local filename, start, stop = self.source:match("^(.-)[(%d+):(%d+)]$")
|
||||
self.source = Location(filename, tonumber(start), tonumber(stop))
|
||||
end
|
||||
for i = 1, select("#", ...) do
|
||||
local x = select(i, ...)
|
||||
assert(type(x) ~= 'table' or getmetatable(x))
|
||||
end
|
||||
self.bits = {
|
||||
...
|
||||
}
|
||||
self.free_vars = { }
|
||||
self.is_value = false
|
||||
end,
|
||||
__base = _base_0,
|
||||
__name = "Lua"
|
||||
@ -227,82 +253,15 @@ do
|
||||
end
|
||||
})
|
||||
_base_0.__class = _class_0
|
||||
Lua = _class_0
|
||||
end
|
||||
do
|
||||
local _class_0
|
||||
local _parent_0 = Lua
|
||||
local _base_0 = {
|
||||
is_statement = false,
|
||||
is_value = true,
|
||||
__tostring = function(self)
|
||||
local buff = { }
|
||||
local _list_0 = self.bits
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local b = _list_0[_index_0]
|
||||
buff[#buff + 1] = tostring(b)
|
||||
end
|
||||
local ret = concat(buff, "")
|
||||
assert(not ret:match(".*table: 0x.*"))
|
||||
return ret
|
||||
end,
|
||||
as_statements = function(self, prefix, suffix)
|
||||
if prefix == nil then
|
||||
prefix = ""
|
||||
end
|
||||
if suffix == nil then
|
||||
suffix = ";"
|
||||
end
|
||||
local bits = {
|
||||
prefix,
|
||||
unpack(self.bits)
|
||||
}
|
||||
bits[#bits + 1] = suffix
|
||||
return Lua(self.source, unpack(bits))
|
||||
end,
|
||||
parenthesize = function(self)
|
||||
self:prepend("(")
|
||||
return self:append(")")
|
||||
end
|
||||
}
|
||||
_base_0.__index = _base_0
|
||||
setmetatable(_base_0, _parent_0.__base)
|
||||
_class_0 = setmetatable({
|
||||
__init = function(self, source, ...)
|
||||
self.source = source
|
||||
self.bits = {
|
||||
...
|
||||
}
|
||||
end,
|
||||
__base = _base_0,
|
||||
__name = "LuaValue",
|
||||
__parent = _parent_0
|
||||
}, {
|
||||
__index = function(cls, name)
|
||||
local val = rawget(_base_0, name)
|
||||
if val == nil then
|
||||
local parent = rawget(cls, "__parent")
|
||||
if parent then
|
||||
return parent[name]
|
||||
end
|
||||
else
|
||||
return val
|
||||
end
|
||||
end,
|
||||
__call = function(cls, ...)
|
||||
local _self_0 = setmetatable({}, _base_0)
|
||||
cls.__init(_self_0, ...)
|
||||
return _self_0
|
||||
end
|
||||
})
|
||||
_base_0.__class = _class_0
|
||||
if _parent_0.__inherited then
|
||||
_parent_0.__inherited(_parent_0, _class_0)
|
||||
local self = _class_0
|
||||
self.Value = function(...)
|
||||
local lua = Lua(...)
|
||||
lua.is_value = true
|
||||
return lua
|
||||
end
|
||||
LuaValue = _class_0
|
||||
Lua = _class_0
|
||||
end
|
||||
return {
|
||||
Lua = Lua,
|
||||
LuaValue = LuaValue,
|
||||
Location = Location
|
||||
}
|
||||
|
87
lua_obj.moon
87
lua_obj.moon
@ -1,12 +1,12 @@
|
||||
{:insert, :remove, :concat} = table
|
||||
immutable = require 'immutable'
|
||||
local Lua, LuaValue, Location
|
||||
local Lua, Location
|
||||
export LINE_STARTS
|
||||
|
||||
Location = immutable {"filename","start","stop"}, {
|
||||
name:"Location"
|
||||
__new: (filename, start, stop)=>
|
||||
assert(type(filename) == 'string' and type(start) == 'number' and type(stop) == 'number')
|
||||
--assert(type(filename) == 'string' and type(start) == 'number' and type(stop) == 'number')
|
||||
return filename, start, stop or start
|
||||
__tostring: => "Location(\"#{@filename}\", #{@start}, #{@stop})"
|
||||
__lt: (other)=>
|
||||
@ -22,12 +22,13 @@ Location = immutable {"filename","start","stop"}, {
|
||||
get_text: => FILE_CACHE[@filename]\sub(@start,@stop)
|
||||
get_line_number: =>
|
||||
-- TODO: do a binary search if this is actually slow, which I doubt
|
||||
line_starts = LINE_STARTS[FILE_CACHE[@filename]]
|
||||
src = FILE_CACHE[@filename]
|
||||
line_starts = LINE_STARTS[src]
|
||||
start_line = 1
|
||||
while (line_starts[start_line+1] or (#src+1)) <= @start
|
||||
while (line_starts[start_line+1] or math.huge) <= @start
|
||||
start_line += 1
|
||||
stop_line = start_line
|
||||
while (line_starts[stop_line+1] or (#src+1)) <= @stop
|
||||
while (line_starts[stop_line+1] or math.huge) <= @stop
|
||||
stop_line += 1
|
||||
return start_line, stop_line
|
||||
get_line: => "#{@filename}:#{@get_line_number!}"
|
||||
@ -39,27 +40,41 @@ Location = immutable {"filename","start","stop"}, {
|
||||
}
|
||||
|
||||
class Lua
|
||||
is_statement: true
|
||||
is_value: false
|
||||
|
||||
new: (@source, ...)=>
|
||||
if type(@source) == 'string'
|
||||
filename,start,stop = @source\match("^(.-)[(%d+):(%d+)]$")
|
||||
@source = Location(filename, tonumber(start), tonumber(stop))
|
||||
for i=1,select("#",...)
|
||||
x = select(i,...)
|
||||
assert(type(x) != 'table' or getmetatable(x))
|
||||
@bits = {...}
|
||||
@free_vars = {}
|
||||
@is_value = false
|
||||
|
||||
add_free_vars: (free_vars)=>
|
||||
@Value = (...)->
|
||||
lua = Lua(...)
|
||||
lua.is_value = true
|
||||
return lua
|
||||
|
||||
clone: =>
|
||||
copy = Lua(@source, {unpack(@bits)})
|
||||
copy.is_value = @is_value
|
||||
for k,v in pairs @free_vars
|
||||
copy.free_vars[k] = v
|
||||
return copy
|
||||
|
||||
add_free_vars: (...)=>
|
||||
seen = {[v]:true for v in *@free_vars}
|
||||
for var in *free_vars
|
||||
for i=1,select("#",...)
|
||||
var = select(i, ...)
|
||||
unless seen[var]
|
||||
@free_vars[#@free_vars+1] = var
|
||||
seen[var] = true
|
||||
|
||||
as_statements: => self
|
||||
convert_to_statements: (prefix="", suffix=";")=>
|
||||
unless @is_value
|
||||
return
|
||||
if prefix != ""
|
||||
@prepend prefix
|
||||
if suffix != ""
|
||||
@append suffix
|
||||
|
||||
declare_locals: (skip={})=>
|
||||
if next(skip) == 1
|
||||
@ -75,10 +90,9 @@ class Lua
|
||||
buff = {}
|
||||
for i,b in ipairs @bits
|
||||
buff[#buff+1] = tostring(b)
|
||||
if i < #@bits and type(b) != 'string' and b.is_statement
|
||||
if i < #@bits and type(b) != 'string' and not b.is_value
|
||||
buff[#buff+1] = "\n"
|
||||
ret = concat(buff, "")
|
||||
assert(not ret\match(".*table: 0x.*"))
|
||||
return ret
|
||||
|
||||
__len: =>
|
||||
@ -87,24 +101,24 @@ class Lua
|
||||
len += #b
|
||||
return len
|
||||
|
||||
__add: (other)=>
|
||||
Lua(nil, self, other)
|
||||
|
||||
__concat: (other)=>
|
||||
Lua(nil, self, other)
|
||||
|
||||
append: (...)=>
|
||||
n = select("#",...)
|
||||
bits = @bits
|
||||
for i=1,n
|
||||
x = select(i,...)
|
||||
assert(type(x) != 'table' or getmetatable(x))
|
||||
bits[#bits+1] = select(i, ...)
|
||||
|
||||
prepend: (...)=>
|
||||
n = select("#",...)
|
||||
bits = @bits
|
||||
for i=#bits+n,n+1,-1
|
||||
x = select(i,...)
|
||||
assert(type(x) != 'table' or getmetatable(x))
|
||||
bits[i] = bits[i-n]
|
||||
for i=1,n
|
||||
x = select(i,...)
|
||||
assert(type(x) != 'table' or getmetatable(x))
|
||||
bits[i] = select(i, ...)
|
||||
|
||||
make_offset_table: (lua_chunkname)=>
|
||||
@ -130,28 +144,11 @@ class Lua
|
||||
walk self
|
||||
return lua_str, metadata
|
||||
|
||||
class LuaValue extends Lua
|
||||
is_statement: false
|
||||
is_value: true
|
||||
|
||||
new: (@source, ...)=>
|
||||
@bits = {...}
|
||||
|
||||
__tostring: =>
|
||||
buff = {}
|
||||
for b in *@bits
|
||||
buff[#buff+1] = tostring(b)
|
||||
ret = concat(buff, "")
|
||||
assert(not ret\match(".*table: 0x.*"))
|
||||
return ret
|
||||
|
||||
as_statements: (prefix="", suffix=";")=>
|
||||
bits = {prefix, unpack @bits}
|
||||
bits[#bits+1] = suffix
|
||||
return Lua(@source, unpack(bits))
|
||||
|
||||
parenthesize: =>
|
||||
@prepend "("
|
||||
@append ")"
|
||||
if @is_value
|
||||
@prepend "("
|
||||
@append ")"
|
||||
else
|
||||
error "Cannot parenthesize lua statements"
|
||||
|
||||
return {:Lua, :LuaValue, :Location}
|
||||
return {:Lua, :Location}
|
||||
|
517
nomsu.lua
517
nomsu.lua
@ -7,6 +7,9 @@ P, R, V, S, Cg, C, Cp, B, Cmt = lpeg.P, lpeg.R, lpeg.V, lpeg.S, lpeg.Cg, lpeg.C,
|
||||
local utils = require('utils')
|
||||
local new_uuid = require('uuid')
|
||||
local immutable = require('immutable')
|
||||
Tuple = immutable(nil, {
|
||||
name = "Tuple"
|
||||
})
|
||||
local repr, stringify, min, max, equivalent, set, is_list, sum
|
||||
repr, stringify, min, max, equivalent, set, is_list, sum = utils.repr, utils.stringify, utils.min, utils.max, utils.equivalent, utils.set, utils.is_list, utils.sum
|
||||
local colors = setmetatable({ }, {
|
||||
@ -27,10 +30,10 @@ do
|
||||
insert, remove, concat = _obj_0.insert, _obj_0.remove, _obj_0.concat
|
||||
end
|
||||
local debug_getinfo = debug.getinfo
|
||||
local Lua, LuaValue, Location
|
||||
local Lua, Location
|
||||
do
|
||||
local _obj_0 = require("lua_obj")
|
||||
Lua, LuaValue, Location = _obj_0.Lua, _obj_0.LuaValue, _obj_0.Location
|
||||
Lua, Location = _obj_0.Lua, _obj_0.Location
|
||||
end
|
||||
FILE_CACHE = setmetatable({ }, {
|
||||
__index = function(self, filename)
|
||||
@ -85,53 +88,7 @@ do
|
||||
end
|
||||
end
|
||||
end
|
||||
local Types = { }
|
||||
local type_tostring
|
||||
type_tostring = function(self)
|
||||
return tostring(self.name) .. "(" .. tostring(repr(self.value)) .. ")"
|
||||
end
|
||||
local type_with_value
|
||||
type_with_value = function(self, value)
|
||||
return getmetatable(self)(self.source, value)
|
||||
end
|
||||
local Tuple = immutable(nil, {
|
||||
name = "Tuple"
|
||||
})
|
||||
local _list_0 = {
|
||||
"File",
|
||||
"Nomsu",
|
||||
"Block",
|
||||
"List",
|
||||
"FunctionCall",
|
||||
"Text",
|
||||
"Dict",
|
||||
"Number",
|
||||
"Word",
|
||||
"Var",
|
||||
"Comment",
|
||||
"IndexChain"
|
||||
}
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local t = _list_0[_index_0]
|
||||
Types[t] = immutable({
|
||||
"value",
|
||||
"source"
|
||||
}, {
|
||||
type = t,
|
||||
name = t,
|
||||
__tostring = type_tostring,
|
||||
with_value = type_with_value
|
||||
})
|
||||
end
|
||||
Types.DictEntry = immutable({
|
||||
"key",
|
||||
"value"
|
||||
}, {
|
||||
name = "DictEntry"
|
||||
})
|
||||
Types.is_node = function(n)
|
||||
return type(n) == 'userdata' and n.type
|
||||
end
|
||||
local Types = require("nomsu_tree")
|
||||
local NOMSU_DEFS
|
||||
do
|
||||
local _with_0 = { }
|
||||
@ -376,7 +333,8 @@ do
|
||||
local tree = self:parse(nomsu_code, filename)
|
||||
assert(tree, "Failed to parse: " .. tostring(nomsu_code))
|
||||
assert(tree.type == "File", "Attempt to run non-file: " .. tostring(tree.type))
|
||||
local lua = self:tree_to_lua(tree):as_statements()
|
||||
local lua = self:tree_to_lua(tree)
|
||||
lua:convert_to_statements()
|
||||
lua:declare_locals()
|
||||
lua:prepend("-- File: " .. tostring(filename) .. "\n")
|
||||
return self:run_lua(lua, filename .. ".lua")
|
||||
@ -511,7 +469,7 @@ do
|
||||
end
|
||||
local nomsu = inline_expression(tok.value)
|
||||
return nomsu and "(: " .. tostring(nomsu) .. ")"
|
||||
elseif "FunctionCall" == _exp_0 then
|
||||
elseif "Action" == _exp_0 then
|
||||
local buff = ""
|
||||
for i, bit in ipairs(tok.value) do
|
||||
if bit.type == "Word" then
|
||||
@ -529,7 +487,7 @@ do
|
||||
buff = buff .. " "
|
||||
end
|
||||
buff = buff .. (function()
|
||||
if bit.type == "FunctionCall" then
|
||||
if bit.type == "Action" then
|
||||
return "(" .. nomsu .. ")"
|
||||
else
|
||||
return nomsu
|
||||
@ -540,9 +498,9 @@ do
|
||||
return buff
|
||||
elseif "IndexChain" == _exp_0 then
|
||||
local bits = { }
|
||||
local _list_1 = tok.value
|
||||
for _index_0 = 1, #_list_1 do
|
||||
local bit = _list_1[_index_0]
|
||||
local _list_0 = tok.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local bit = _list_0[_index_0]
|
||||
local nomsu = inline_expression(bit)
|
||||
if not (nomsu) then
|
||||
return nil
|
||||
@ -552,9 +510,9 @@ do
|
||||
return concat(bits, ".")
|
||||
elseif "List" == _exp_0 then
|
||||
local bits = { }
|
||||
local _list_1 = tok.value
|
||||
for _index_0 = 1, #_list_1 do
|
||||
local bit = _list_1[_index_0]
|
||||
local _list_0 = tok.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local bit = _list_0[_index_0]
|
||||
local nomsu = inline_expression(bit)
|
||||
if not (nomsu) then
|
||||
return nil
|
||||
@ -564,9 +522,9 @@ do
|
||||
return "[" .. concat(bits, ", ") .. "]"
|
||||
elseif "Dict" == _exp_0 then
|
||||
local bits = { }
|
||||
local _list_1 = tok.value
|
||||
for _index_0 = 1, #_list_1 do
|
||||
local bit = _list_1[_index_0]
|
||||
local _list_0 = tok.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local bit = _list_0[_index_0]
|
||||
local key_nomsu
|
||||
if bit.key.type == "Word" then
|
||||
key_nomsu = bit.key.value
|
||||
@ -576,7 +534,7 @@ do
|
||||
if not (key_nomsu) then
|
||||
return nil
|
||||
end
|
||||
if bit.key.type == "FunctionCall" then
|
||||
if bit.key.type == "Action" then
|
||||
key_nomsu = "(" .. key_nomsu .. ")"
|
||||
end
|
||||
local value_nomsu = inline_expression(bit.value)
|
||||
@ -588,9 +546,9 @@ do
|
||||
return "{" .. concat(bits, ", ") .. "}"
|
||||
elseif "Text" == _exp_0 then
|
||||
local buff = '"'
|
||||
local _list_1 = tok.value
|
||||
for _index_0 = 1, #_list_1 do
|
||||
local bit = _list_1[_index_0]
|
||||
local _list_0 = tok.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local bit = _list_0[_index_0]
|
||||
if type(bit) == 'string' then
|
||||
if bit:find("\n") then
|
||||
return nil
|
||||
@ -636,9 +594,9 @@ do
|
||||
local _exp_0 = tok.type
|
||||
if "Block" == _exp_0 then
|
||||
local buff = ":"
|
||||
local _list_1 = tok.value
|
||||
for _index_0 = 1, #_list_1 do
|
||||
local line = _list_1[_index_0]
|
||||
local _list_0 = tok.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local line = _list_0[_index_0]
|
||||
nomsu = expression(line)
|
||||
if not (nomsu) then
|
||||
return nil
|
||||
@ -646,7 +604,7 @@ do
|
||||
buff = buff .. ("\n " .. self:indent(nomsu))
|
||||
end
|
||||
return buff
|
||||
elseif "FunctionCall" == _exp_0 then
|
||||
elseif "Action" == _exp_0 then
|
||||
nomsu = expression(tok)
|
||||
if not (nomsu) then
|
||||
return nil
|
||||
@ -657,9 +615,9 @@ do
|
||||
elseif "List" == _exp_0 then
|
||||
local buff = "[..]"
|
||||
local line = "\n "
|
||||
local _list_1 = tok.value
|
||||
for _index_0 = 1, #_list_1 do
|
||||
local bit = _list_1[_index_0]
|
||||
local _list_0 = tok.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local bit = _list_0[_index_0]
|
||||
nomsu = inline_expression(bit)
|
||||
if line ~= "\n " and #line + #", " + #nomsu > max_line then
|
||||
buff = buff .. line
|
||||
@ -685,14 +643,14 @@ do
|
||||
elseif "Dict" == _exp_0 then
|
||||
local buff = "{..}"
|
||||
local line = "\n "
|
||||
local _list_1 = tok.value
|
||||
for _index_0 = 1, #_list_1 do
|
||||
local bit = _list_1[_index_0]
|
||||
local _list_0 = tok.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local bit = _list_0[_index_0]
|
||||
local key_nomsu = inline_expression(bit.key)
|
||||
if not (key_nomsu) then
|
||||
return nil
|
||||
end
|
||||
if bit.key.type == "FunctionCall" then
|
||||
if bit.key.type == "Action" then
|
||||
key_nomsu = "(" .. key_nomsu .. ")"
|
||||
end
|
||||
local value_nomsu = inline_expression(bit.value)
|
||||
@ -714,9 +672,9 @@ do
|
||||
return buff
|
||||
elseif "Text" == _exp_0 then
|
||||
local buff = '".."\n '
|
||||
local _list_1 = tok.value
|
||||
for _index_0 = 1, #_list_1 do
|
||||
local bit = _list_1[_index_0]
|
||||
local _list_0 = tok.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local bit = _list_0[_index_0]
|
||||
if type(bit) == 'string' then
|
||||
buff = buff .. bit:gsub("\\", "\\\\"):gsub("\n", "\n ")
|
||||
else
|
||||
@ -758,7 +716,7 @@ do
|
||||
local _exp_0 = tok.type
|
||||
if "Block" == _exp_0 then
|
||||
if #tok.value == 1 then
|
||||
if tok.value[1].type == "FunctionCall" then
|
||||
if tok.value[1].type == "Action" then
|
||||
nomsu = inline_expression(tok.value[1])
|
||||
else
|
||||
nomsu = noeol_expression(tok.value[1])
|
||||
@ -768,7 +726,7 @@ do
|
||||
end
|
||||
end
|
||||
return noeol_expression(tok)
|
||||
elseif "FunctionCall" == _exp_0 then
|
||||
elseif "Action" == _exp_0 then
|
||||
local buff = ""
|
||||
for i, bit in ipairs(tok.value) do
|
||||
if bit.type == "Word" then
|
||||
@ -780,7 +738,7 @@ do
|
||||
else
|
||||
nomsu = inline_expression(bit)
|
||||
if nomsu and #nomsu < max_line then
|
||||
if bit.type == "FunctionCall" then
|
||||
if bit.type == "Action" then
|
||||
nomsu = "(" .. nomsu .. ")"
|
||||
end
|
||||
else
|
||||
@ -788,7 +746,7 @@ do
|
||||
if not (nomsu) then
|
||||
return nil
|
||||
end
|
||||
if bit.type == "FunctionCall" then
|
||||
if bit.type == "Action" then
|
||||
nomsu = "(..)\n " .. self:indent(nomsu)
|
||||
end
|
||||
if i < #tok.value then
|
||||
@ -804,9 +762,9 @@ do
|
||||
return buff
|
||||
elseif "File" == _exp_0 then
|
||||
local lines = { }
|
||||
local _list_1 = tree.value
|
||||
for _index_0 = 1, #_list_1 do
|
||||
local line = _list_1[_index_0]
|
||||
local _list_0 = tree.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local line = _list_0[_index_0]
|
||||
nomsu = expression(line)
|
||||
if not (nomsu) then
|
||||
error("Failed to produce output for:\n" .. tostring(colored.yellow(line.source:get_text())), 0)
|
||||
@ -870,309 +828,7 @@ do
|
||||
end
|
||||
end,
|
||||
tree_to_lua = function(self, tree)
|
||||
assert(tree, "No tree provided.")
|
||||
if not Types.is_node(tree) then
|
||||
error("Invalid tree")
|
||||
end
|
||||
local _exp_0 = tree.type
|
||||
if "File" == _exp_0 then
|
||||
if #tree.value == 1 then
|
||||
return self:tree_to_lua(tree.value[1])
|
||||
end
|
||||
local file_lua = Lua(tree.source)
|
||||
local declared_locals = { }
|
||||
for i, line in ipairs(tree.value) do
|
||||
local line_lua = self:tree_to_lua(line)
|
||||
if not line_lua then
|
||||
error("No lua produced by " .. tostring(repr(line)), 0)
|
||||
end
|
||||
line_lua = line_lua:as_statements()
|
||||
if i < #tree.value then
|
||||
file_lua:append("\n")
|
||||
end
|
||||
file_lua:append(line_lua)
|
||||
end
|
||||
file_lua:declare_locals()
|
||||
return file_lua
|
||||
elseif "Comment" == _exp_0 then
|
||||
return Lua(tree.source, "--" .. tree.value:gsub("\n", "\n--") .. "\n")
|
||||
elseif "Nomsu" == _exp_0 then
|
||||
return Lua(tree.source, "nomsu:parse(", tree.source:get_text(), ", ", repr(tree.source.filename), ")")
|
||||
elseif "Block" == _exp_0 then
|
||||
local block_lua = Lua(tree.source)
|
||||
for i, arg in ipairs(tree.value) do
|
||||
local lua = self:tree_to_lua(arg)
|
||||
if i == 1 and #tree.value == 1 and lua.is_value then
|
||||
return lua
|
||||
end
|
||||
lua = lua:as_statements()
|
||||
if i < #tree.value then
|
||||
block_lua:append("\n")
|
||||
end
|
||||
block_lua:append(lua)
|
||||
end
|
||||
return block_lua
|
||||
elseif "FunctionCall" == _exp_0 then
|
||||
insert(self.compilestack, tree)
|
||||
local stub = self:tree_to_stub(tree)
|
||||
local action = rawget(self.environment.ACTIONS, stub)
|
||||
local metadata = self.action_metadata[action]
|
||||
if metadata and metadata.compile_time then
|
||||
local args
|
||||
do
|
||||
local _accum_0 = { }
|
||||
local _len_0 = 1
|
||||
local _list_1 = tree.value
|
||||
for _index_0 = 1, #_list_1 do
|
||||
local arg = _list_1[_index_0]
|
||||
if arg.type ~= "Word" then
|
||||
_accum_0[_len_0] = arg
|
||||
_len_0 = _len_0 + 1
|
||||
end
|
||||
end
|
||||
args = _accum_0
|
||||
end
|
||||
if metadata and metadata.arg_orders then
|
||||
local new_args
|
||||
do
|
||||
local _accum_0 = { }
|
||||
local _len_0 = 1
|
||||
local _list_1 = metadata.arg_orders[stub]
|
||||
for _index_0 = 1, #_list_1 do
|
||||
local p = _list_1[_index_0]
|
||||
_accum_0[_len_0] = args[p - 1]
|
||||
_len_0 = _len_0 + 1
|
||||
end
|
||||
new_args = _accum_0
|
||||
end
|
||||
args = new_args
|
||||
end
|
||||
local lua = action(tree, unpack(args))
|
||||
remove(self.compilestack)
|
||||
return lua
|
||||
elseif not metadata and self.__class.math_patt:match(stub) then
|
||||
local lua = LuaValue(tree.source)
|
||||
for i, tok in ipairs(tree.value) do
|
||||
if tok.type == "Word" then
|
||||
lua:append(tok.value)
|
||||
else
|
||||
local tok_lua = self:tree_to_lua(tok)
|
||||
if not (tok_lua.is_value) then
|
||||
local src = tok.source:get_text()
|
||||
error("non-expression value inside math expression: " .. tostring(colored.yellow(src)))
|
||||
end
|
||||
lua:append(tok_lua)
|
||||
end
|
||||
if i < #tree.value then
|
||||
lua:append(" ")
|
||||
end
|
||||
end
|
||||
remove(self.compilestack)
|
||||
lua:parenthesize()
|
||||
return lua
|
||||
end
|
||||
local args = { }
|
||||
for i, tok in ipairs(tree.value) do
|
||||
local _continue_0 = false
|
||||
repeat
|
||||
if tok.type == "Word" then
|
||||
_continue_0 = true
|
||||
break
|
||||
end
|
||||
local arg_lua = self:tree_to_lua(tok)
|
||||
if not (arg_lua.is_value) then
|
||||
local line, src = tok.source:get_line(), tok.source:get_text()
|
||||
error(tostring(line) .. ": Cannot use:\n" .. tostring(colored.yellow(src)) .. "\nas an argument to " .. tostring(stub) .. ", since it's not an expression, it produces: " .. tostring(repr(arg_lua)), 0)
|
||||
end
|
||||
insert(args, arg_lua)
|
||||
_continue_0 = true
|
||||
until true
|
||||
if not _continue_0 then
|
||||
break
|
||||
end
|
||||
end
|
||||
if metadata and metadata.arg_orders then
|
||||
do
|
||||
local _accum_0 = { }
|
||||
local _len_0 = 1
|
||||
local _list_1 = metadata.arg_orders[stub]
|
||||
for _index_0 = 1, #_list_1 do
|
||||
local p = _list_1[_index_0]
|
||||
_accum_0[_len_0] = args[p]
|
||||
_len_0 = _len_0 + 1
|
||||
end
|
||||
args = _accum_0
|
||||
end
|
||||
end
|
||||
local lua = LuaValue(tree.source, "ACTIONS[" .. tostring(repr(stub)) .. "](")
|
||||
for i, arg in ipairs(args) do
|
||||
lua:append(arg)
|
||||
if i < #args then
|
||||
lua:append(", ")
|
||||
end
|
||||
end
|
||||
lua:append(")")
|
||||
remove(self.compilestack)
|
||||
return lua
|
||||
elseif "Text" == _exp_0 then
|
||||
local lua = LuaValue(tree.source)
|
||||
local string_buffer = ""
|
||||
local _list_1 = tree.value
|
||||
for _index_0 = 1, #_list_1 do
|
||||
local _continue_0 = false
|
||||
repeat
|
||||
local bit = _list_1[_index_0]
|
||||
if type(bit) == "string" then
|
||||
string_buffer = string_buffer .. bit
|
||||
_continue_0 = true
|
||||
break
|
||||
end
|
||||
if string_buffer ~= "" then
|
||||
if #lua.bits > 0 then
|
||||
lua:append("..")
|
||||
end
|
||||
lua:append(repr(string_buffer))
|
||||
string_buffer = ""
|
||||
end
|
||||
local bit_lua = self:tree_to_lua(bit)
|
||||
if not (bit_lua.is_value) then
|
||||
local line, src = bit.source:get_line(), bit.source:get_text()
|
||||
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(bit)) .. " as a string interpolation value, since it's not an expression.", 0)
|
||||
end
|
||||
if #lua.bits > 0 then
|
||||
lua:append("..")
|
||||
end
|
||||
if bit.type ~= "Text" then
|
||||
bit_lua = LuaValue(bit.source, "stringify(", bit_lua, ")")
|
||||
end
|
||||
lua:append(bit_lua)
|
||||
_continue_0 = true
|
||||
until true
|
||||
if not _continue_0 then
|
||||
break
|
||||
end
|
||||
end
|
||||
if string_buffer ~= "" or #lua.bits == 0 then
|
||||
if #lua.bits > 0 then
|
||||
lua:append("..")
|
||||
end
|
||||
lua:append(repr(string_buffer))
|
||||
end
|
||||
if #lua.bits > 1 then
|
||||
lua:parenthesize()
|
||||
end
|
||||
return lua
|
||||
elseif "IndexChain" == _exp_0 then
|
||||
local lua = self:tree_to_lua(tree.value[1])
|
||||
if not (lua.is_value) then
|
||||
local line, src = tree.value[1].source:get_line(), tree.value[1].source:get_text()
|
||||
error(tostring(line) .. ": Cannot index " .. tostring(colored.yellow(src)) .. ", since it's not an expression.", 0)
|
||||
end
|
||||
local last_char = tostring(lua):sub(1, 1)
|
||||
if last_char == "}" or last_char == '"' then
|
||||
lua:parenthesize()
|
||||
end
|
||||
for i = 2, #tree.value do
|
||||
local _continue_0 = false
|
||||
repeat
|
||||
local key = tree.value[i]
|
||||
if key.type == 'Text' and #key.value == 1 and type(key.value[1]) == 'string' and key.value[1]:match("^[a-zA-Z_][a-zA-Z0-9_]$") then
|
||||
lua:append("." .. tostring(key.value[1]))
|
||||
_continue_0 = true
|
||||
break
|
||||
end
|
||||
local key_lua = self:tree_to_lua(key)
|
||||
if not (key_lua.is_value) then
|
||||
local line, src = key.source:get_line(), key.source:get_text()
|
||||
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(src)) .. " as an index, since it's not an expression.", 0)
|
||||
end
|
||||
if tostring(key_lua):sub(1, 1) == '[' then
|
||||
lua:append("[ ", key_lua, "]")
|
||||
else
|
||||
lua:append("[", key_lua, "]")
|
||||
end
|
||||
_continue_0 = true
|
||||
until true
|
||||
if not _continue_0 then
|
||||
break
|
||||
end
|
||||
end
|
||||
return lua
|
||||
elseif "List" == _exp_0 then
|
||||
local lua = LuaValue(tree.source, "{")
|
||||
local line_length = 0
|
||||
for i, item in ipairs(tree.value) do
|
||||
local item_lua = self:tree_to_lua(item)
|
||||
if not (item_lua.is_value) then
|
||||
local line, src = item.source:get_line(), item.source:get_text()
|
||||
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(src)) .. " as a list item, since it's not an expression.", 0)
|
||||
end
|
||||
lua:append(item_lua)
|
||||
local newlines, last_line = tostring(item_lua):match("^(.-)([^\n]*)$")
|
||||
if #newlines > 0 then
|
||||
line_length = #last_line
|
||||
else
|
||||
line_length = line_length + #last_line
|
||||
end
|
||||
if i < #tree.value then
|
||||
if line_length >= 80 then
|
||||
lua:append(",\n")
|
||||
line_length = 0
|
||||
else
|
||||
lua:append(", ")
|
||||
line_length = line_length + 2
|
||||
end
|
||||
end
|
||||
end
|
||||
lua:append("}")
|
||||
return lua
|
||||
elseif "Dict" == _exp_0 then
|
||||
local lua = LuaValue(tree.source, "{")
|
||||
local line_length = 0
|
||||
for i, entry in ipairs(tree.value) do
|
||||
local key_lua = self:tree_to_lua(entry.key)
|
||||
if not (key_lua.is_value) then
|
||||
local line, src = key.source:get_line(), key.source:get_text()
|
||||
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(src)) .. " as a dict key, since it's not an expression.", 0)
|
||||
end
|
||||
local value_lua = self:tree_to_lua(entry.value)
|
||||
if not (value_lua.is_value) then
|
||||
local line, src = value.source:get_line(), value.source:get_text()
|
||||
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(src)) .. " as a dict value, since it's not an expression.", 0)
|
||||
end
|
||||
local key_str = tostring(key_lua):match([=[["']([a-zA-Z_][a-zA-Z0-9_]*)['"]]=])
|
||||
if key_str then
|
||||
lua:append(key_str, "=", value_lua)
|
||||
elseif tostring(key_lua):sub(1, 1) == "[" then
|
||||
lua:append("[ ", key_lua, "]=", value_lua)
|
||||
else
|
||||
lua:append("[", key_lua, "]=", value_lua)
|
||||
end
|
||||
local newlines, last_line = ("[" .. tostring(key_lua) .. "=" .. tostring(value_lua)):match("^(.-)([^\n]*)$")
|
||||
if #newlines > 0 then
|
||||
line_length = #last_line
|
||||
else
|
||||
line_length = line_length + #last_line
|
||||
end
|
||||
if i < #tree.value then
|
||||
if line_length >= 80 then
|
||||
lua:append(",\n")
|
||||
line_length = 0
|
||||
else
|
||||
lua:append(", ")
|
||||
line_length = line_length + 2
|
||||
end
|
||||
end
|
||||
end
|
||||
lua:append("}")
|
||||
return lua
|
||||
elseif "Number" == _exp_0 then
|
||||
return LuaValue(tree.source, tostring(tree.value))
|
||||
elseif "Var" == _exp_0 then
|
||||
return LuaValue(tree.source, self:var_to_lua_identifier(tree.value))
|
||||
else
|
||||
return error("Unknown/unimplemented thingy: " .. tostring(tree.type), 0)
|
||||
end
|
||||
return tree:as_lua(self)
|
||||
end,
|
||||
walk_tree = function(self, tree, depth)
|
||||
if depth == nil then
|
||||
@ -1183,16 +839,16 @@ do
|
||||
return
|
||||
end
|
||||
local _exp_0 = tree.type
|
||||
if "List" == _exp_0 or "File" == _exp_0 or "Block" == _exp_0 or "FunctionCall" == _exp_0 or "Text" == _exp_0 or "IndexChain" == _exp_0 then
|
||||
local _list_1 = tree.value
|
||||
for _index_0 = 1, #_list_1 do
|
||||
local v = _list_1[_index_0]
|
||||
if "List" == _exp_0 or "File" == _exp_0 or "Block" == _exp_0 or "Action" == _exp_0 or "Text" == _exp_0 or "IndexChain" == _exp_0 then
|
||||
local _list_0 = tree.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local v = _list_0[_index_0]
|
||||
self:walk_tree(v, depth + 1)
|
||||
end
|
||||
elseif "Dict" == _exp_0 then
|
||||
local _list_1 = tree.value
|
||||
for _index_0 = 1, #_list_1 do
|
||||
local e = _list_1[_index_0]
|
||||
local _list_0 = tree.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local e = _list_0[_index_0]
|
||||
self:walk_tree(e.key, depth + 1)
|
||||
self:walk_tree(e.value, depth + 1)
|
||||
end
|
||||
@ -1236,7 +892,7 @@ do
|
||||
return replacement
|
||||
end
|
||||
local _exp_0 = tree.type
|
||||
if "File" == _exp_0 or "Nomsu" == _exp_0 or "Block" == _exp_0 or "List" == _exp_0 or "FunctionCall" == _exp_0 or "Text" == _exp_0 or "IndexChain" == _exp_0 then
|
||||
if "File" == _exp_0 or "Nomsu" == _exp_0 or "Block" == _exp_0 or "List" == _exp_0 or "Action" == _exp_0 or "Text" == _exp_0 or "IndexChain" == _exp_0 then
|
||||
local new_values, is_changed = { }, false
|
||||
for i, old_value in ipairs(tree.value) do
|
||||
local new_value = type(old_value) ~= "string" and self:tree_map(old_value, fn) or nil
|
||||
@ -1281,15 +937,15 @@ do
|
||||
end)
|
||||
end,
|
||||
tree_to_stub = function(self, tree)
|
||||
if tree.type ~= "FunctionCall" then
|
||||
if tree.type ~= "Action" then
|
||||
error("Tried to get stub from non-functioncall tree: " .. tostring(tree.type), 0)
|
||||
end
|
||||
return concat((function()
|
||||
local _accum_0 = { }
|
||||
local _len_0 = 1
|
||||
local _list_1 = tree.value
|
||||
for _index_0 = 1, #_list_1 do
|
||||
local t = _list_1[_index_0]
|
||||
local _list_0 = tree.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local t = _list_0[_index_0]
|
||||
_accum_0[_len_0] = (t.type == "Word" and t.value or "%")
|
||||
_len_0 = _len_0 + 1
|
||||
end
|
||||
@ -1297,15 +953,15 @@ do
|
||||
end)(), " ")
|
||||
end,
|
||||
tree_to_named_stub = function(self, tree)
|
||||
if tree.type ~= "FunctionCall" then
|
||||
if tree.type ~= "Action" then
|
||||
error("Tried to get stub from non-functioncall tree: " .. tostring(tree.type), 0)
|
||||
end
|
||||
return concat((function()
|
||||
local _accum_0 = { }
|
||||
local _len_0 = 1
|
||||
local _list_1 = tree.value
|
||||
for _index_0 = 1, #_list_1 do
|
||||
local t = _list_1[_index_0]
|
||||
local _list_0 = tree.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local t = _list_0[_index_0]
|
||||
_accum_0[_len_0] = (t.type == "Word" and t.value or "%" .. tostring(t.value))
|
||||
_len_0 = _len_0 + 1
|
||||
end
|
||||
@ -1367,19 +1023,20 @@ do
|
||||
end
|
||||
local nomsu = self
|
||||
self:define_compile_action("immediately %block", get_line_no(), function(self, _block)
|
||||
local lua = nomsu:tree_to_lua(_block):as_statements()
|
||||
local lua = nomsu:tree_to_lua(_block)
|
||||
lua:convert_to_statements()
|
||||
lua:declare_locals()
|
||||
nomsu:run_lua(lua)
|
||||
return Lua(self.source, "if IMMEDIATE then\n", lua, "\nend")
|
||||
end)
|
||||
self:define_compile_action("Lua %code", get_line_no(), function(self, _code)
|
||||
self:define_compile_action("Lua %source %code", get_line_no(), function(self, _source, _code)
|
||||
if _code.type ~= "Text" then
|
||||
return LuaValue(self.source, "Lua(", repr(_code.source), ", ", nomsu:tree_to_lua(_code), ")")
|
||||
return Lua.Value(_source, "Lua(", repr(_code.source), ", ", nomsu:tree_to_lua(_code), ")")
|
||||
end
|
||||
local lua = LuaValue(self.source, "Lua(", repr(_code.source))
|
||||
local _list_1 = _code.value
|
||||
for _index_0 = 1, #_list_1 do
|
||||
local bit = _list_1[_index_0]
|
||||
local lua = Lua.Value(_source, "Lua(", repr(_code.source))
|
||||
local _list_0 = _code.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local bit = _list_0[_index_0]
|
||||
lua:append(", ")
|
||||
if type(bit) == "string" then
|
||||
lua:append(repr(bit))
|
||||
@ -1395,14 +1052,14 @@ do
|
||||
lua:append(")")
|
||||
return lua
|
||||
end)
|
||||
self:define_compile_action("LuaValue %code", get_line_no(), function(self, _code)
|
||||
self:define_compile_action("Lua value %source %code", get_line_no(), function(self, _source, _code)
|
||||
if _code.type ~= "Text" then
|
||||
return LuaValue(self.source, "LuaValue(", repr(_code.source), ", ", nomsu:tree_to_lua(_code), ")")
|
||||
return Lua.Value(_source, "Lua.Value(", repr(_code.source), ", ", nomsu:tree_to_lua(_code), ")")
|
||||
end
|
||||
local lua = LuaValue(self.source, "LuaValue(", repr(_code.source))
|
||||
local _list_1 = _code.value
|
||||
for _index_0 = 1, #_list_1 do
|
||||
local bit = _list_1[_index_0]
|
||||
local lua = Lua.Value(_source, "Lua.Value(", repr(_code.source))
|
||||
local _list_0 = _code.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local bit = _list_0[_index_0]
|
||||
lua:append(", ")
|
||||
if type(bit) == "string" then
|
||||
lua:append(repr(bit))
|
||||
@ -1420,12 +1077,12 @@ do
|
||||
end)
|
||||
self:define_compile_action("lua> %code", get_line_no(), function(self, _code)
|
||||
if _code.type ~= "Text" then
|
||||
return LuaValue(self.source, "nomsu:run_lua(", nomsu:tree_to_lua(_code), ")")
|
||||
return Lua.Value(self.source, "nomsu:run_lua(", nomsu:tree_to_lua(_code), ")")
|
||||
end
|
||||
local lua = Lua(_code.source)
|
||||
local _list_1 = _code.value
|
||||
for _index_0 = 1, #_list_1 do
|
||||
local bit = _list_1[_index_0]
|
||||
local _list_0 = _code.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local bit = _list_0[_index_0]
|
||||
if type(bit) == "string" then
|
||||
lua:append(bit)
|
||||
else
|
||||
@ -1441,12 +1098,12 @@ do
|
||||
end)
|
||||
self:define_compile_action("=lua %code", get_line_no(), function(self, _code)
|
||||
if _code.type ~= "Text" then
|
||||
return LuaValue(self.source, "nomsu:run_lua(", nomsu:tree_to_lua(_code), ")")
|
||||
return Lua.Value(self.source, "nomsu:run_lua(", nomsu:tree_to_lua(_code), ")")
|
||||
end
|
||||
local lua = LuaValue(self.source)
|
||||
local _list_1 = _code.value
|
||||
for _index_0 = 1, #_list_1 do
|
||||
local bit = _list_1[_index_0]
|
||||
local lua = Lua.Value(self.source)
|
||||
local _list_0 = _code.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local bit = _list_0[_index_0]
|
||||
if type(bit) == "string" then
|
||||
lua:append(bit)
|
||||
else
|
||||
@ -1461,7 +1118,7 @@ do
|
||||
return lua
|
||||
end)
|
||||
self:define_compile_action("!! code location !!", get_line_no(), function(self)
|
||||
return LuaValue(self.source, repr(self.source))
|
||||
return Lua.Value(self.source, repr(tostring(self.source)))
|
||||
end)
|
||||
self:define_action("run file %filename", get_line_no(), function(_filename)
|
||||
return nomsu:run_file(_filename)
|
||||
@ -1469,7 +1126,7 @@ do
|
||||
return self:define_compile_action("use %filename", get_line_no(), function(self, _filename)
|
||||
local filename = nomsu:tree_to_value(_filename)
|
||||
nomsu:use_file(filename)
|
||||
return LuaValue(self.source, "nomsu:use_file(" .. tostring(repr(filename)) .. ")")
|
||||
return Lua.Value(self.source, "nomsu:use_file(" .. tostring(repr(filename)) .. ")")
|
||||
end)
|
||||
end
|
||||
}
|
||||
@ -1548,7 +1205,6 @@ do
|
||||
end
|
||||
self.environment.Tuple = Tuple
|
||||
self.environment.Lua = Lua
|
||||
self.environment.LuaValue = LuaValue
|
||||
self.environment.Location = Location
|
||||
self.environment.ACTIONS = setmetatable({ }, {
|
||||
__index = function(self, key)
|
||||
@ -1572,7 +1228,6 @@ do
|
||||
_base_0.__class = _class_0
|
||||
local self = _class_0
|
||||
_chunk_counter = 0
|
||||
self.math_patt = re.compile([[ "%" (" " [*/^+-] " %")+ ]])
|
||||
self.unescape_string = function(self, str)
|
||||
return Cs(((P("\\\\") / "\\") + (P("\\\"") / '"') + NOMSU_DEFS.escaped_char + P(1)) ^ 0):match(str)
|
||||
end
|
||||
|
296
nomsu.moon
296
nomsu.moon
@ -18,12 +18,14 @@ lpeg.setmaxstack 10000
|
||||
utils = require 'utils'
|
||||
new_uuid = require 'uuid'
|
||||
immutable = require 'immutable'
|
||||
export Tuple
|
||||
Tuple = immutable(nil, {name:"Tuple"})
|
||||
{:repr, :stringify, :min, :max, :equivalent, :set, :is_list, :sum} = utils
|
||||
colors = setmetatable({}, {__index:->""})
|
||||
colored = setmetatable({}, {__index:(_,color)-> ((msg)-> colors[color]..tostring(msg or '')..colors.reset)})
|
||||
{:insert, :remove, :concat} = table
|
||||
debug_getinfo = debug.getinfo
|
||||
{:Lua, :LuaValue, :Location} = require "lua_obj"
|
||||
{:Lua, :Location} = require "lua_obj"
|
||||
|
||||
-- TODO:
|
||||
-- consider non-linear codegen, rather than doing thunks for things like comprehensions
|
||||
@ -88,16 +90,7 @@ do
|
||||
-- Can't use this because it breaks some LPEG stuff
|
||||
--STRING_METATABLE.__mul = (other)=> string.rep(@, other)
|
||||
|
||||
Types = {}
|
||||
type_tostring = =>
|
||||
"#{@name}(#{repr(@value)})"
|
||||
type_with_value = (value)=> getmetatable(self)(@source, value)
|
||||
Tuple = immutable(nil, {name:"Tuple"})
|
||||
for t in *{"File", "Nomsu", "Block", "List", "FunctionCall", "Text", "Dict", "Number", "Word", "Var", "Comment", "IndexChain"}
|
||||
Types[t] = immutable({"value","source"}, {type:t, name:t, __tostring:type_tostring, with_value:type_with_value})
|
||||
Types.DictEntry = immutable({"key","value"}, {name:"DictEntry"})
|
||||
Types.is_node = (n)->
|
||||
type(n) == 'userdata' and n.type
|
||||
Types = require "nomsu_tree"
|
||||
|
||||
NOMSU_DEFS = with {}
|
||||
-- Newline supports either windows-style CR+LF or unix-style LF
|
||||
@ -218,7 +211,6 @@ class NomsuCompiler
|
||||
for k,v in pairs(Types) do @environment[k] = v
|
||||
@environment.Tuple = Tuple
|
||||
@environment.Lua = Lua
|
||||
@environment.LuaValue = LuaValue
|
||||
@environment.Location = Location
|
||||
@environment.ACTIONS = setmetatable({}, {__index:(key)=>
|
||||
error("Attempt to run undefined action: #{key}", 0)
|
||||
@ -305,7 +297,8 @@ class NomsuCompiler
|
||||
tree = @parse(nomsu_code, filename)
|
||||
assert tree, "Failed to parse: #{nomsu_code}"
|
||||
assert tree.type == "File", "Attempt to run non-file: #{tree.type}"
|
||||
lua = @tree_to_lua(tree)\as_statements!
|
||||
lua = @tree_to_lua(tree)
|
||||
lua\convert_to_statements!
|
||||
lua\declare_locals!
|
||||
lua\prepend "-- File: #{filename}\n"
|
||||
return @run_lua(lua, filename..".lua")
|
||||
@ -404,7 +397,7 @@ class NomsuCompiler
|
||||
if #tok.value > 1 then return nil
|
||||
nomsu = inline_expression tok.value
|
||||
return nomsu and "(: #{nomsu})"
|
||||
when "FunctionCall"
|
||||
when "Action"
|
||||
buff = ""
|
||||
for i,bit in ipairs tok.value
|
||||
if bit.type == "Word"
|
||||
@ -416,7 +409,7 @@ class NomsuCompiler
|
||||
return nil unless nomsu
|
||||
unless i == 1 or bit.type == "Block"
|
||||
buff ..= " "
|
||||
buff ..= if bit.type == "FunctionCall"
|
||||
buff ..= if bit.type == "Action"
|
||||
"("..nomsu..")"
|
||||
else nomsu
|
||||
return buff
|
||||
@ -441,7 +434,7 @@ class NomsuCompiler
|
||||
bit.key.value
|
||||
else inline_expression bit.key
|
||||
return nil unless key_nomsu
|
||||
if bit.key.type == "FunctionCall"
|
||||
if bit.key.type == "Action"
|
||||
key_nomsu = "("..key_nomsu..")"
|
||||
value_nomsu = inline_expression bit.value
|
||||
return nil unless value_nomsu
|
||||
@ -482,7 +475,7 @@ class NomsuCompiler
|
||||
return nil unless nomsu
|
||||
buff ..= "\n "..@indent(nomsu)
|
||||
return buff
|
||||
when "FunctionCall"
|
||||
when "Action"
|
||||
nomsu = expression(tok)
|
||||
return nil unless nomsu
|
||||
return "(..)\n "..@indent(nomsu)
|
||||
@ -515,7 +508,7 @@ class NomsuCompiler
|
||||
for bit in *tok.value
|
||||
key_nomsu = inline_expression bit.key
|
||||
return nil unless key_nomsu
|
||||
if bit.key.type == "FunctionCall"
|
||||
if bit.key.type == "Action"
|
||||
key_nomsu = "("..key_nomsu..")"
|
||||
value_nomsu = inline_expression bit.value
|
||||
if value_nomsu and #key_nomsu + #value_nomsu < max_line
|
||||
@ -560,14 +553,14 @@ class NomsuCompiler
|
||||
switch tok.type
|
||||
when "Block"
|
||||
if #tok.value == 1
|
||||
nomsu = if tok.value[1].type == "FunctionCall"
|
||||
nomsu = if tok.value[1].type == "Action"
|
||||
inline_expression(tok.value[1])
|
||||
else
|
||||
noeol_expression(tok.value[1])
|
||||
if nomsu and #(nomsu\match("[^\n]*")) < max_line
|
||||
return ": "..nomsu
|
||||
return noeol_expression(tok)
|
||||
when "FunctionCall"
|
||||
when "Action"
|
||||
-- The hard task
|
||||
buff = ""
|
||||
for i,bit in ipairs tok.value
|
||||
@ -579,12 +572,12 @@ class NomsuCompiler
|
||||
else
|
||||
nomsu = inline_expression(bit)
|
||||
if nomsu and #nomsu < max_line
|
||||
if bit.type == "FunctionCall"
|
||||
if bit.type == "Action"
|
||||
nomsu = "("..nomsu..")"
|
||||
else
|
||||
nomsu = expression(bit)
|
||||
return nil unless nomsu
|
||||
if bit.type == "FunctionCall"
|
||||
if bit.type == "Action"
|
||||
nomsu = "(..)\n "..@indent(nomsu)
|
||||
if i < #tok.value
|
||||
nomsu ..= "\n.."
|
||||
@ -636,238 +629,14 @@ class NomsuCompiler
|
||||
else
|
||||
error("Unsupported value_to_nomsu type: #{type(value)}", 0)
|
||||
|
||||
@math_patt: re.compile [[ "%" (" " [*/^+-] " %")+ ]]
|
||||
tree_to_lua: (tree)=>
|
||||
-- Return <lua code for value>, <additional lua code>
|
||||
assert tree, "No tree provided."
|
||||
if not Types.is_node(tree)
|
||||
--error("Invalid tree: #{repr(tree)}", 0)
|
||||
error("Invalid tree")
|
||||
switch tree.type
|
||||
when "File"
|
||||
if #tree.value == 1
|
||||
return @tree_to_lua(tree.value[1])
|
||||
file_lua = Lua(tree.source)
|
||||
declared_locals = {}
|
||||
for i, line in ipairs tree.value
|
||||
line_lua = @tree_to_lua line
|
||||
if not line_lua
|
||||
error("No lua produced by #{repr line}", 0)
|
||||
line_lua = line_lua\as_statements!
|
||||
if i < #tree.value
|
||||
file_lua\append "\n"
|
||||
|
||||
file_lua\append line_lua
|
||||
file_lua\declare_locals!
|
||||
return file_lua
|
||||
|
||||
when "Comment"
|
||||
return Lua(tree.source, "--"..tree.value\gsub("\n","\n--").."\n")
|
||||
|
||||
when "Nomsu"
|
||||
--return Lua(tree.source, repr(tree.value))
|
||||
return Lua(tree.source, "nomsu:parse(",tree.source\get_text!,", ",repr(tree.source.filename),")")
|
||||
return tree\as_lua(self)
|
||||
|
||||
when "Block"
|
||||
block_lua = Lua(tree.source)
|
||||
for i,arg in ipairs tree.value
|
||||
lua = @tree_to_lua arg
|
||||
if i == 1 and #tree.value == 1 and lua.is_value
|
||||
return lua
|
||||
lua = lua\as_statements!
|
||||
if i < #tree.value
|
||||
block_lua\append "\n"
|
||||
block_lua\append lua
|
||||
return block_lua
|
||||
|
||||
when "FunctionCall"
|
||||
insert @compilestack, tree
|
||||
|
||||
stub = @tree_to_stub tree
|
||||
action = rawget(@environment.ACTIONS, stub)
|
||||
|
||||
metadata = @action_metadata[action]
|
||||
if metadata and metadata.compile_time
|
||||
args = [arg for arg in *tree.value when arg.type != "Word"]
|
||||
-- Force all compile-time actions to take a tree location
|
||||
if metadata and metadata.arg_orders
|
||||
new_args = [args[p-1] for p in *metadata.arg_orders[stub]]
|
||||
args = new_args
|
||||
lua = action(tree, unpack(args))
|
||||
remove @compilestack
|
||||
return lua
|
||||
elseif not metadata and @@math_patt\match(stub)
|
||||
-- This is a bit of a hack, but this code handles arbitrarily complex
|
||||
-- math expressions like 2*x + 3^2 without having to define a single
|
||||
-- action for every possibility.
|
||||
lua = LuaValue(tree.source)
|
||||
for i,tok in ipairs tree.value
|
||||
if tok.type == "Word"
|
||||
lua\append tok.value
|
||||
else
|
||||
tok_lua = @tree_to_lua(tok)
|
||||
unless tok_lua.is_value
|
||||
src = tok.source\get_text!
|
||||
error("non-expression value inside math expression: #{colored.yellow src}")
|
||||
lua\append tok_lua
|
||||
if i < #tree.value
|
||||
lua\append " "
|
||||
remove @compilestack
|
||||
lua\parenthesize!
|
||||
return lua
|
||||
|
||||
args = {}
|
||||
for i, tok in ipairs tree.value
|
||||
if tok.type == "Word" then continue
|
||||
arg_lua = @tree_to_lua(tok)
|
||||
unless arg_lua.is_value
|
||||
line, src = tok.source\get_line!, tok.source\get_text!
|
||||
error "#{line}: Cannot use:\n#{colored.yellow src}\nas an argument to #{stub}, since it's not an expression, it produces: #{repr arg_lua}", 0
|
||||
insert args, arg_lua
|
||||
|
||||
if metadata and metadata.arg_orders
|
||||
args = [args[p] for p in *metadata.arg_orders[stub]]
|
||||
|
||||
lua = LuaValue(tree.source, "ACTIONS[#{repr stub}](")
|
||||
for i, arg in ipairs args
|
||||
lua\append arg
|
||||
if i < #args then lua\append ", "
|
||||
lua\append ")"
|
||||
remove @compilestack
|
||||
return lua
|
||||
|
||||
when "Text"
|
||||
lua = LuaValue(tree.source)
|
||||
string_buffer = ""
|
||||
for bit in *tree.value
|
||||
if type(bit) == "string"
|
||||
string_buffer ..= bit
|
||||
continue
|
||||
if string_buffer ~= ""
|
||||
if #lua.bits > 0 then lua\append ".."
|
||||
lua\append repr(string_buffer)
|
||||
string_buffer = ""
|
||||
bit_lua = @tree_to_lua bit
|
||||
unless bit_lua.is_value
|
||||
line, src = bit.source\get_line!, bit.source\get_text!
|
||||
error "#{line}: Cannot use #{colored.yellow bit} as a string interpolation value, since it's not an expression.", 0
|
||||
if #lua.bits > 0 then lua\append ".."
|
||||
if bit.type != "Text"
|
||||
bit_lua = LuaValue(bit.source, "stringify(",bit_lua,")")
|
||||
lua\append bit_lua
|
||||
|
||||
if string_buffer ~= "" or #lua.bits == 0
|
||||
if #lua.bits > 0 then lua\append ".."
|
||||
lua\append repr(string_buffer)
|
||||
|
||||
if #lua.bits > 1
|
||||
lua\parenthesize!
|
||||
return lua
|
||||
|
||||
when "IndexChain"
|
||||
lua = @tree_to_lua tree.value[1]
|
||||
unless lua.is_value
|
||||
line, src = tree.value[1].source\get_line!, tree.value[1].source\get_text!
|
||||
error "#{line}: Cannot index #{colored.yellow src}, since it's not an expression.", 0
|
||||
last_char = tostring(lua)\sub(1,1)
|
||||
if last_char == "}" or last_char == '"'
|
||||
lua\parenthesize!
|
||||
|
||||
for i=2,#tree.value
|
||||
key = tree.value[i]
|
||||
if key.type == 'Text' and #key.value == 1 and type(key.value[1]) == 'string' and key.value[1]\match("^[a-zA-Z_][a-zA-Z0-9_]$")
|
||||
lua\append ".#{key.value[1]}"
|
||||
continue
|
||||
key_lua = @tree_to_lua key
|
||||
unless key_lua.is_value
|
||||
line, src = key.source\get_line!, key.source\get_text!
|
||||
error "#{line}: Cannot use #{colored.yellow src} as an index, since it's not an expression.", 0
|
||||
-- NOTE: this *must* use a space after the [ to avoid freaking out
|
||||
-- Lua's parser if the inner expression is a long string. Lua
|
||||
-- parses x[[[y]]] as x("[y]"), not as x["y"]
|
||||
if tostring(key_lua)\sub(1,1) == '['
|
||||
lua\append "[ ",key_lua,"]"
|
||||
else
|
||||
lua\append "[",key_lua,"]"
|
||||
return lua
|
||||
|
||||
when "List"
|
||||
lua = LuaValue tree.source, "{"
|
||||
line_length = 0
|
||||
for i, item in ipairs tree.value
|
||||
item_lua = @tree_to_lua item
|
||||
unless item_lua.is_value
|
||||
line, src = item.source\get_line!, item.source\get_text!
|
||||
error "#{line}: Cannot use #{colored.yellow src} as a list item, since it's not an expression.", 0
|
||||
lua\append item_lua
|
||||
newlines, last_line = tostring(item_lua)\match("^(.-)([^\n]*)$")
|
||||
if #newlines > 0
|
||||
line_length = #last_line
|
||||
else
|
||||
line_length += #last_line
|
||||
if i < #tree.value
|
||||
if line_length >= 80
|
||||
lua\append ",\n"
|
||||
line_length = 0
|
||||
else
|
||||
lua\append ", "
|
||||
line_length += 2
|
||||
lua\append "}"
|
||||
return lua
|
||||
|
||||
when "Dict"
|
||||
lua = LuaValue tree.source, "{"
|
||||
line_length = 0
|
||||
for i, entry in ipairs tree.value
|
||||
key_lua = @tree_to_lua entry.key
|
||||
unless key_lua.is_value
|
||||
line, src = key.source\get_line!, key.source\get_text!
|
||||
error "#{line}: Cannot use #{colored.yellow src} as a dict key, since it's not an expression.", 0
|
||||
value_lua = @tree_to_lua entry.value
|
||||
unless value_lua.is_value
|
||||
line, src = value.source\get_line!, value.source\get_text!
|
||||
error "#{line}: Cannot use #{colored.yellow src} as a dict value, since it's not an expression.", 0
|
||||
key_str = tostring(key_lua)\match([=[["']([a-zA-Z_][a-zA-Z0-9_]*)['"]]=])
|
||||
if key_str
|
||||
lua\append key_str,"=",value_lua
|
||||
elseif tostring(key_lua)\sub(1,1) == "["
|
||||
-- NOTE: this *must* use a space after the [ to avoid freaking out
|
||||
-- Lua's parser if the inner expression is a long string. Lua
|
||||
-- parses x[[[y]]] as x("[y]"), not as x["y"]
|
||||
lua\append "[ ",key_lua,"]=",value_lua
|
||||
else
|
||||
lua\append "[",key_lua,"]=",value_lua
|
||||
|
||||
-- TODO: maybe make this more accurate? It's only a heuristic, so eh...
|
||||
newlines, last_line = ("[#{key_lua}=#{value_lua}")\match("^(.-)([^\n]*)$")
|
||||
if #newlines > 0
|
||||
line_length = #last_line
|
||||
else
|
||||
line_length += #last_line
|
||||
if i < #tree.value
|
||||
if line_length >= 80
|
||||
lua\append ",\n"
|
||||
line_length = 0
|
||||
else
|
||||
lua\append ", "
|
||||
line_length += 2
|
||||
lua\append "}"
|
||||
return lua
|
||||
|
||||
when "Number"
|
||||
return LuaValue(tree.source, tostring(tree.value))
|
||||
|
||||
when "Var"
|
||||
return LuaValue(tree.source, @var_to_lua_identifier(tree.value))
|
||||
|
||||
else
|
||||
error("Unknown/unimplemented thingy: #{tree.type}", 0)
|
||||
|
||||
walk_tree: (tree, depth=0)=>
|
||||
coroutine.yield(tree, depth)
|
||||
return unless Types.is_node(tree)
|
||||
switch tree.type
|
||||
when "List", "File", "Block", "FunctionCall", "Text", "IndexChain"
|
||||
when "List", "File", "Block", "Action", "Text", "IndexChain"
|
||||
for v in *tree.value
|
||||
@walk_tree(v, depth+1)
|
||||
when "Dict"
|
||||
@ -919,7 +688,7 @@ class NomsuCompiler
|
||||
if replacement != nil
|
||||
return replacement
|
||||
switch tree.type
|
||||
when "File", "Nomsu", "Block", "List", "FunctionCall", "Text", "IndexChain"
|
||||
when "File", "Nomsu", "Block", "List", "Action", "Text", "IndexChain"
|
||||
new_values, is_changed = {}, false
|
||||
for i,old_value in ipairs(tree.value)
|
||||
new_value = type(old_value) != "string" and @tree_map(old_value, fn) or nil
|
||||
@ -956,11 +725,11 @@ class NomsuCompiler
|
||||
return replacements[id]
|
||||
|
||||
tree_to_stub: (tree)=>
|
||||
if tree.type != "FunctionCall" then error "Tried to get stub from non-functioncall tree: #{tree.type}", 0
|
||||
if tree.type != "Action" then error "Tried to get stub from non-functioncall tree: #{tree.type}", 0
|
||||
return concat([(t.type == "Word" and t.value or "%") for t in *tree.value], " ")
|
||||
|
||||
tree_to_named_stub: (tree)=>
|
||||
if tree.type != "FunctionCall" then error "Tried to get stub from non-functioncall tree: #{tree.type}", 0
|
||||
if tree.type != "Action" then error "Tried to get stub from non-functioncall tree: #{tree.type}", 0
|
||||
return concat([(t.type == "Word" and t.value or "%#{t.value}") for t in *tree.value], " ")
|
||||
|
||||
stub_defs = {
|
||||
@ -1009,16 +778,17 @@ class NomsuCompiler
|
||||
get_line_no = -> "nomsu.moon:#{debug_getinfo(2).currentline}"
|
||||
nomsu = self
|
||||
@define_compile_action "immediately %block", get_line_no!, (_block)=>
|
||||
lua = nomsu\tree_to_lua(_block)\as_statements!
|
||||
lua = nomsu\tree_to_lua(_block)
|
||||
lua\convert_to_statements!
|
||||
lua\declare_locals!
|
||||
nomsu\run_lua(lua)
|
||||
return Lua(@source, "if IMMEDIATE then\n", lua, "\nend")
|
||||
|
||||
@define_compile_action "Lua %code", get_line_no!, (_code)=>
|
||||
@define_compile_action "Lua %source %code", get_line_no!, (_source, _code)=>
|
||||
if _code.type != "Text"
|
||||
return LuaValue(@source, "Lua(", repr(_code.source),", ",nomsu\tree_to_lua(_code),")")
|
||||
return Lua.Value(_source, "Lua(", repr(_code.source),", ",nomsu\tree_to_lua(_code),")")
|
||||
|
||||
lua = LuaValue(@source, "Lua(", repr(_code.source))
|
||||
lua = Lua.Value(_source, "Lua(", repr(_code.source))
|
||||
for bit in *_code.value
|
||||
lua\append ", "
|
||||
if type(bit) == "string"
|
||||
@ -1032,11 +802,11 @@ class NomsuCompiler
|
||||
lua\append ")"
|
||||
return lua
|
||||
|
||||
@define_compile_action "LuaValue %code", get_line_no!, (_code)=>
|
||||
@define_compile_action "Lua value %source %code", get_line_no!, (_source, _code)=>
|
||||
if _code.type != "Text"
|
||||
return LuaValue(@source, "LuaValue(", repr(_code.source),", ",nomsu\tree_to_lua(_code),")")
|
||||
return Lua.Value(_source, "Lua.Value(", repr(_code.source),", ",nomsu\tree_to_lua(_code),")")
|
||||
|
||||
lua = LuaValue(@source, "LuaValue(", repr(_code.source))
|
||||
lua = Lua.Value(_source, "Lua.Value(", repr(_code.source))
|
||||
for bit in *_code.value
|
||||
lua\append ", "
|
||||
if type(bit) == "string"
|
||||
@ -1052,7 +822,7 @@ class NomsuCompiler
|
||||
|
||||
@define_compile_action "lua> %code", get_line_no!, (_code)=>
|
||||
if _code.type != "Text"
|
||||
return LuaValue(@source, "nomsu:run_lua(",nomsu\tree_to_lua(_code),")")
|
||||
return Lua.Value(@source, "nomsu:run_lua(",nomsu\tree_to_lua(_code),")")
|
||||
|
||||
lua = Lua(_code.source)
|
||||
for bit in *_code.value
|
||||
@ -1068,9 +838,9 @@ class NomsuCompiler
|
||||
|
||||
@define_compile_action "=lua %code", get_line_no!, (_code)=>
|
||||
if _code.type != "Text"
|
||||
return LuaValue(@source, "nomsu:run_lua(",nomsu\tree_to_lua(_code),")")
|
||||
return Lua.Value(@source, "nomsu:run_lua(",nomsu\tree_to_lua(_code),")")
|
||||
|
||||
lua = LuaValue(@source)
|
||||
lua = Lua.Value(@source)
|
||||
for bit in *_code.value
|
||||
if type(bit) == "string"
|
||||
lua\append bit
|
||||
@ -1083,7 +853,7 @@ class NomsuCompiler
|
||||
return lua
|
||||
|
||||
@define_compile_action "!! code location !!", get_line_no!, =>
|
||||
return LuaValue(@source, repr(@source))
|
||||
return Lua.Value(@source, repr(tostring(@source)))
|
||||
|
||||
@define_action "run file %filename", get_line_no!, (_filename)->
|
||||
return nomsu\run_file(_filename)
|
||||
@ -1091,7 +861,7 @@ class NomsuCompiler
|
||||
@define_compile_action "use %filename", get_line_no!, (_filename)=>
|
||||
filename = nomsu\tree_to_value(_filename)
|
||||
nomsu\use_file(filename)
|
||||
return LuaValue(@source, "nomsu:use_file(#{repr filename})")
|
||||
return Lua.Value(@source, "nomsu:use_file(#{repr filename})")
|
||||
|
||||
-- Only run this code if this file was run directly with command line arguments, and not require()'d:
|
||||
if arg and debug_getinfo(2).func != require
|
||||
|
26
nomsu.peg
26
nomsu.peg
@ -7,7 +7,7 @@ file (File):
|
||||
|
||||
shebang: "#!" [^%nl]* %nl
|
||||
|
||||
statement: functioncall / expression
|
||||
statement: action / expression
|
||||
|
||||
indented_block (Block):
|
||||
{| (":" / "(..)")? indent
|
||||
@ -17,27 +17,27 @@ indented_block (Block):
|
||||
|
||||
inline_nomsu (Nomsu): "\" noindex_inline_expression
|
||||
indented_nomsu (Nomsu):
|
||||
"\" (noindex_inline_expression / (":" %ws* (inline_functioncall / inline_expression) eol) / indented_expression)
|
||||
"\" (noindex_inline_expression / (":" %ws* (inline_action / inline_expression) eol) / indented_expression)
|
||||
|
||||
index_chain (IndexChain):
|
||||
{| noindex_inline_expression ("." ((({} ({|{%operator / (!number plain_word)}|} -> Tuple) {}) -> Text) / noindex_inline_expression))+ |} -> Tuple
|
||||
|
||||
noindex_inline_expression:
|
||||
number / variable / inline_text / inline_list / inline_dict / inline_nomsu
|
||||
/ ("(" %ws* (inline_functioncall / inline_expression) %ws* ")")
|
||||
/ ("(" %ws* (inline_action / inline_expression) %ws* ")")
|
||||
|
||||
inline_expression:
|
||||
index_chain / noindex_inline_expression
|
||||
indented_expression:
|
||||
indented_text / indented_nomsu / indented_list / indented_dict / indented_block
|
||||
expression:
|
||||
inline_expression / (":" %ws* (inline_functioncall / inline_expression) eol) / indented_expression
|
||||
inline_expression / (":" %ws* (inline_action / inline_expression) eol) / indented_expression
|
||||
|
||||
-- Function calls need at least one word in them
|
||||
inline_functioncall (FunctionCall):
|
||||
inline_action (Action):
|
||||
{| (inline_expression %ws*)* word (%ws* (inline_expression / word))*
|
||||
(%ws* ":" %ws* (inline_functioncall / inline_expression))?|} -> Tuple
|
||||
functioncall (FunctionCall):
|
||||
(%ws* ":" %ws* (inline_action / inline_expression))?|} -> Tuple
|
||||
action (Action):
|
||||
{| (expression (dotdot / %ws*))* word ((dotdot / %ws*) (expression / word))* |} -> Tuple
|
||||
|
||||
word (Word): { %operator / (!number plain_word) }
|
||||
@ -60,13 +60,13 @@ indented_text (Text):
|
||||
inline_text_interpolation:
|
||||
"\" (
|
||||
variable / inline_list / inline_dict / inline_text
|
||||
/ ("(" %ws* (inline_functioncall / inline_expression) %ws* ")")
|
||||
/ ("(" %ws* (inline_action / inline_expression) %ws* ")")
|
||||
)
|
||||
text_interpolation:
|
||||
inline_text_interpolation /
|
||||
("\" (
|
||||
variable / inline_list / inline_dict / inline_text
|
||||
/ ("(" %ws* (inline_functioncall / inline_expression) %ws* ")")
|
||||
/ ("(" %ws* (inline_action / inline_expression) %ws* ")")
|
||||
/ (%ws* (block_comment / line_comment)? nodent "..")
|
||||
/ (indented_text %nl+ %nodent "..")
|
||||
/ ((indented_list / indented_block) nodent "..")
|
||||
@ -87,9 +87,9 @@ indented_list (List):
|
||||
|} -> Tuple)
|
||||
(dedent / (("" -> "Error while parsing list") => error))
|
||||
list_line:
|
||||
((functioncall / expression) !comma)
|
||||
((action / expression) !comma)
|
||||
/ (inline_list_item (comma list_line?)?)
|
||||
inline_list_item: inline_functioncall / inline_expression
|
||||
inline_list_item: inline_action / inline_expression
|
||||
|
||||
inline_dict (Dict):
|
||||
!('{..}')
|
||||
@ -100,10 +100,10 @@ indented_dict (Dict):
|
||||
|} -> Tuple)
|
||||
(dedent / (("" -> "Error while parsing dict") => error))
|
||||
dict_line:
|
||||
((dict_key %ws* ":" %ws* (functioncall / expression)) -> DictEntry !comma)
|
||||
((dict_key %ws* ":" %ws* (action / expression)) -> DictEntry !comma)
|
||||
/ (inline_dict_item (comma dict_line?)?)
|
||||
inline_dict_item:
|
||||
(dict_key %ws* ":" %ws* (inline_functioncall / inline_expression)) -> DictEntry
|
||||
(dict_key %ws* ":" %ws* (inline_action / inline_expression)) -> DictEntry
|
||||
dict_key:
|
||||
(({} ({|{%operator / (!number plain_word)}|} -> Tuple) {}) -> Text) / inline_expression
|
||||
|
||||
|
405
nomsu_tree.lua
Normal file
405
nomsu_tree.lua
Normal file
@ -0,0 +1,405 @@
|
||||
local utils = require('utils')
|
||||
local re = require('re')
|
||||
local repr, stringify, min, max, equivalent, set, is_list, sum
|
||||
repr, stringify, min, max, equivalent, set, is_list, sum = utils.repr, utils.stringify, utils.min, utils.max, utils.equivalent, utils.set, utils.is_list, utils.sum
|
||||
local immutable = require('immutable')
|
||||
local insert, remove, concat
|
||||
do
|
||||
local _obj_0 = table
|
||||
insert, remove, concat = _obj_0.insert, _obj_0.remove, _obj_0.concat
|
||||
end
|
||||
local Lua, Location
|
||||
do
|
||||
local _obj_0 = require("lua_obj")
|
||||
Lua, Location = _obj_0.Lua, _obj_0.Location
|
||||
end
|
||||
local common_methods = {
|
||||
__tostring = function(self)
|
||||
return tostring(self.name) .. "(" .. tostring(repr(self.value)) .. ")"
|
||||
end,
|
||||
with_value = function(self, value)
|
||||
return getmetatable(self)(value, self.source)
|
||||
end
|
||||
}
|
||||
local fields = {
|
||||
"value",
|
||||
"source"
|
||||
}
|
||||
local Types = { }
|
||||
Types.DictEntry = immutable({
|
||||
"key",
|
||||
"value"
|
||||
}, {
|
||||
name = "DictEntry"
|
||||
})
|
||||
Types.is_node = function(n)
|
||||
return type(n) == 'userdata' and getmetatable(n) and Types[n.type] == getmetatable(n)
|
||||
end
|
||||
local Tree
|
||||
Tree = function(name, methods)
|
||||
do
|
||||
methods.__tostring = function(self)
|
||||
return tostring(self.name) .. "(" .. tostring(repr(self.value)) .. ")"
|
||||
end
|
||||
methods.with_value = function(self, value)
|
||||
return getmetatable(self)(value, self.source)
|
||||
end
|
||||
methods.type = name
|
||||
methods.name = name
|
||||
end
|
||||
Types[name] = immutable({
|
||||
"value",
|
||||
"source"
|
||||
}, methods)
|
||||
end
|
||||
Tree("File", {
|
||||
as_lua = function(self, nomsu)
|
||||
if #self.value == 1 then
|
||||
return self.value[1]:as_lua(nomsu)
|
||||
end
|
||||
local declared_locals = { }
|
||||
local lua = Lua(self.source)
|
||||
for i, line in ipairs(self.value) do
|
||||
local line_lua = line:as_lua(nomsu)
|
||||
if not line_lua then
|
||||
error("No lua produced by " .. tostring(repr(line)), 0)
|
||||
end
|
||||
if i < #self.value then
|
||||
lua:append("\n")
|
||||
end
|
||||
lua:convert_to_statements()
|
||||
lua:append(line_lua)
|
||||
end
|
||||
lua:declare_locals()
|
||||
return lua
|
||||
end
|
||||
})
|
||||
Tree("Nomsu", {
|
||||
as_lua = function(self, nomsu)
|
||||
return Lua.Value(self.source, "nomsu:parse(", repr(self.source:get_text()), ", ", repr(self.source.filename), ")")
|
||||
end
|
||||
})
|
||||
Tree("Block", {
|
||||
as_lua = function(self, nomsu)
|
||||
if #self.value == 1 then
|
||||
return self.value[1]:as_lua(nomsu)
|
||||
end
|
||||
local lua = Lua(self.source)
|
||||
for i, line in ipairs(self.value) do
|
||||
local line_lua = line:as_lua(nomsu)
|
||||
if i < #self.value then
|
||||
lua:append("\n")
|
||||
end
|
||||
line_lua:convert_to_statements()
|
||||
lua:append(line_lua)
|
||||
end
|
||||
return lua
|
||||
end
|
||||
})
|
||||
local math_patt = re.compile([[ "%" (" " [*/^+-] " %")+ ]])
|
||||
Tree("Action", {
|
||||
as_lua = function(self, nomsu)
|
||||
local stub = self:get_stub()
|
||||
local action = rawget(nomsu.environment.ACTIONS, stub)
|
||||
local metadata = nomsu.action_metadata[action]
|
||||
if metadata and metadata.compile_time then
|
||||
local args
|
||||
do
|
||||
local _accum_0 = { }
|
||||
local _len_0 = 1
|
||||
local _list_0 = self.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local arg = _list_0[_index_0]
|
||||
if arg.type ~= "Word" then
|
||||
_accum_0[_len_0] = arg
|
||||
_len_0 = _len_0 + 1
|
||||
end
|
||||
end
|
||||
args = _accum_0
|
||||
end
|
||||
if metadata.arg_orders then
|
||||
local new_args
|
||||
do
|
||||
local _accum_0 = { }
|
||||
local _len_0 = 1
|
||||
local _list_0 = metadata.arg_orders[stub]
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local p = _list_0[_index_0]
|
||||
_accum_0[_len_0] = args[p - 1]
|
||||
_len_0 = _len_0 + 1
|
||||
end
|
||||
new_args = _accum_0
|
||||
end
|
||||
args = new_args
|
||||
end
|
||||
return action(Lua(self.source), unpack(args))
|
||||
end
|
||||
local lua = Lua.Value(self.source)
|
||||
if not metadata and math_patt:match(stub) then
|
||||
for i, tok in ipairs(self.value) do
|
||||
if tok.type == "Word" then
|
||||
lua:append(tok.value)
|
||||
else
|
||||
local tok_lua = tok:as_lua(nomsu)
|
||||
if not (tok_lua.is_value) then
|
||||
local src = tok.source:get_text()
|
||||
error("non-expression value inside math expression: " .. tostring(colored.yellow(src)))
|
||||
end
|
||||
lua:append(tok_lua)
|
||||
end
|
||||
if i < #self.value then
|
||||
lua:append(" ")
|
||||
end
|
||||
end
|
||||
lua:parenthesize()
|
||||
return lua
|
||||
end
|
||||
local args = { }
|
||||
for i, tok in ipairs(self.value) do
|
||||
local _continue_0 = false
|
||||
repeat
|
||||
if tok.type == "Word" then
|
||||
_continue_0 = true
|
||||
break
|
||||
end
|
||||
local arg_lua = tok:as_lua(nomsu)
|
||||
if not (arg_lua.is_value) then
|
||||
local line, src = tok.source:get_line(), tok.source:get_text()
|
||||
error(tostring(line) .. ": Cannot use:\n" .. tostring(colored.yellow(src)) .. "\nas an argument to " .. tostring(stub) .. ", since it's not an expression, it produces: " .. tostring(repr(arg_lua)), 0)
|
||||
end
|
||||
insert(args, arg_lua)
|
||||
_continue_0 = true
|
||||
until true
|
||||
if not _continue_0 then
|
||||
break
|
||||
end
|
||||
end
|
||||
if metadata and metadata.arg_orders then
|
||||
do
|
||||
local _accum_0 = { }
|
||||
local _len_0 = 1
|
||||
local _list_0 = metadata.arg_orders[stub]
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local p = _list_0[_index_0]
|
||||
_accum_0[_len_0] = args[p]
|
||||
_len_0 = _len_0 + 1
|
||||
end
|
||||
args = _accum_0
|
||||
end
|
||||
end
|
||||
lua:append("ACTIONS[", repr(stub), "(")
|
||||
for i, arg in ipairs(args) do
|
||||
lua:append(arg)
|
||||
if i < #args then
|
||||
lua:append(", ")
|
||||
end
|
||||
end
|
||||
lua:append(")")
|
||||
return lua
|
||||
end,
|
||||
get_stub = function(self)
|
||||
return concat((function()
|
||||
local _accum_0 = { }
|
||||
local _len_0 = 1
|
||||
local _list_0 = self.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local t = _list_0[_index_0]
|
||||
_accum_0[_len_0] = (t.type == "Word" and t.value or "%")
|
||||
_len_0 = _len_0 + 1
|
||||
end
|
||||
return _accum_0
|
||||
end)(), " ")
|
||||
end
|
||||
})
|
||||
Tree("Text", {
|
||||
as_lua = function(self, nomsu)
|
||||
local lua = Lua.Value(self.source)
|
||||
local string_buffer = ""
|
||||
local _list_0 = self.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local _continue_0 = false
|
||||
repeat
|
||||
local bit = _list_0[_index_0]
|
||||
if type(bit) == "string" then
|
||||
string_buffer = string_buffer .. bit
|
||||
_continue_0 = true
|
||||
break
|
||||
end
|
||||
if string_buffer ~= "" then
|
||||
if #lua.bits > 0 then
|
||||
lua:append("..")
|
||||
end
|
||||
lua:append(repr(string_buffer))
|
||||
string_buffer = ""
|
||||
end
|
||||
local bit_lua = bit:as_lua(nomsu)
|
||||
if not (bit_lua.is_value) then
|
||||
local line, src = bit.source:get_line(), bit.source:get_text()
|
||||
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(bit)) .. " as a string interpolation value, since it's not an expression.", 0)
|
||||
end
|
||||
if #lua.bits > 0 then
|
||||
lua:append("..")
|
||||
end
|
||||
if bit.type ~= "Text" then
|
||||
bit_lua = Lua.Value(bit.source, "stringify(", bit_lua, ")")
|
||||
end
|
||||
lua:append(bit_lua)
|
||||
_continue_0 = true
|
||||
until true
|
||||
if not _continue_0 then
|
||||
break
|
||||
end
|
||||
end
|
||||
if string_buffer ~= "" or #lua.bits == 0 then
|
||||
if #lua.bits > 0 then
|
||||
lua:append("..")
|
||||
end
|
||||
lua:append(repr(string_buffer))
|
||||
end
|
||||
if #lua.bits > 1 then
|
||||
lua:parenthesize()
|
||||
end
|
||||
return lua
|
||||
end
|
||||
})
|
||||
Tree("List", {
|
||||
as_lua = function(self, nomsu)
|
||||
local lua = Lua.Value(self.source, "{")
|
||||
local line_length = 0
|
||||
for i, item in ipairs(self.value) do
|
||||
local item_lua = item:as_lua(nomsu)
|
||||
if not (item_lua.is_value) then
|
||||
local line, src = item.source:get_line(), item.source:get_text()
|
||||
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(src)) .. " as a list item, since it's not an expression.", 0)
|
||||
end
|
||||
lua:append(item_lua)
|
||||
local newlines, last_line = tostring(item_lua):match("^(.-)([^\n]*)$")
|
||||
if #newlines > 0 then
|
||||
line_length = #last_line
|
||||
else
|
||||
line_length = line_length + #last_line
|
||||
end
|
||||
if i < #self.value then
|
||||
if line_length >= 80 then
|
||||
lua:append(",\n")
|
||||
line_length = 0
|
||||
else
|
||||
lua:append(", ")
|
||||
line_length = line_length + 2
|
||||
end
|
||||
end
|
||||
end
|
||||
lua:append("}")
|
||||
return lua
|
||||
end
|
||||
})
|
||||
Tree("Dict", {
|
||||
as_lua = function(self, nomsu)
|
||||
local lua = Lua.Value(self.source, "{")
|
||||
local line_length = 0
|
||||
for i, entry in ipairs(self.value) do
|
||||
local key_lua = entry.key:as_lua(nomsu)
|
||||
if not (key_lua.is_value) then
|
||||
local line, src = key.source:get_line(), key.source:get_text()
|
||||
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(src)) .. " as a dict key, since it's not an expression.", 0)
|
||||
end
|
||||
local value_lua = entry.value:as_lua(nomsu)
|
||||
if not (value_lua.is_value) then
|
||||
local line, src = value.source:get_line(), value.source:get_text()
|
||||
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(src)) .. " as a dict value, since it's not an expression.", 0)
|
||||
end
|
||||
local key_str = tostring(key_lua):match([=[["']([a-zA-Z_][a-zA-Z0-9_]*)['"]]=])
|
||||
if key_str then
|
||||
lua:append(key_str, "=", value_lua)
|
||||
elseif tostring(key_lua):sub(1, 1) == "[" then
|
||||
lua:append("[ ", key_lua, "]=", value_lua)
|
||||
else
|
||||
lua:append("[", key_lua, "]=", value_lua)
|
||||
end
|
||||
local newlines, last_line = ("[" .. tostring(key_lua) .. "=" .. tostring(value_lua)):match("^(.-)([^\n]*)$")
|
||||
if #newlines > 0 then
|
||||
line_length = #last_line
|
||||
else
|
||||
line_length = line_length + #last_line
|
||||
end
|
||||
if i < #self.value then
|
||||
if line_length >= 80 then
|
||||
lua:append(",\n")
|
||||
line_length = 0
|
||||
else
|
||||
lua:append(", ")
|
||||
line_length = line_length + 2
|
||||
end
|
||||
end
|
||||
end
|
||||
lua:append("}")
|
||||
return lua
|
||||
end
|
||||
})
|
||||
Tree("IndexChain", {
|
||||
as_lua = function(self, nomsu)
|
||||
local lua = self.value[1]:as_lua(nomsu)
|
||||
if not (lua.is_value) then
|
||||
local line, src = self.value[1].source:get_line(), self.value[1].source:get_text()
|
||||
error(tostring(line) .. ": Cannot index " .. tostring(colored.yellow(src)) .. ", since it's not an expression.", 0)
|
||||
end
|
||||
local last_char = tostring(lua):sub(-1, -1)
|
||||
if last_char == "}" or last_char == '"' or last_char == "]" then
|
||||
lua:parenthesize()
|
||||
end
|
||||
for i = 2, #self.value do
|
||||
local _continue_0 = false
|
||||
repeat
|
||||
local key = self.value[i]
|
||||
if key.type == 'Text' and #key.value == 1 and type(key.value[1]) == 'string' and key.value[1]:match("^[a-zA-Z_][a-zA-Z0-9_]$") then
|
||||
lua:append("." .. tostring(key.value[1]))
|
||||
_continue_0 = true
|
||||
break
|
||||
end
|
||||
local key_lua = self:tree_to_lua(key)
|
||||
if not (key_lua.is_value) then
|
||||
local line, src = key.source:get_line(), key.source:get_text()
|
||||
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(src)) .. " as an index, since it's not an expression.", 0)
|
||||
end
|
||||
if tostring(key_lua):sub(1, 1) == '[' then
|
||||
lua:append("[ ", key_lua, "]")
|
||||
else
|
||||
lua:append("[", key_lua, "]")
|
||||
end
|
||||
_continue_0 = true
|
||||
until true
|
||||
if not _continue_0 then
|
||||
break
|
||||
end
|
||||
end
|
||||
return lua
|
||||
end
|
||||
})
|
||||
Tree("Number", {
|
||||
as_lua = function(self, nomsu)
|
||||
return Lua.Value(self.source, tostring(self.value))
|
||||
end
|
||||
})
|
||||
Tree("Var", {
|
||||
as_lua = function(self, nomsu)
|
||||
local lua_id = "_" .. (self.value:gsub("%W", function(verboten)
|
||||
if verboten == "_" then
|
||||
return "__"
|
||||
else
|
||||
return ("_%x"):format(verboten:byte())
|
||||
end
|
||||
end))
|
||||
return Lua.Value(self.source, lua_id)
|
||||
end
|
||||
})
|
||||
Tree("Word", {
|
||||
as_lua = function(self, nomsu)
|
||||
return error("Attempt to convert Word to lua")
|
||||
end
|
||||
})
|
||||
Tree("Comment", {
|
||||
as_lua = function(self, nomsu)
|
||||
return Lua(self.source, "--" .. self.value:gsub("\n", "\n--") .. "\n")
|
||||
end
|
||||
})
|
||||
return Types
|
269
nomsu_tree.moon
Normal file
269
nomsu_tree.moon
Normal file
@ -0,0 +1,269 @@
|
||||
-- This file contains the datastructures used to represent parsed Nomsu syntax trees,
|
||||
-- as well as the logic for converting them to Lua code.
|
||||
utils = require 'utils'
|
||||
re = require 're'
|
||||
{:repr, :stringify, :min, :max, :equivalent, :set, :is_list, :sum} = utils
|
||||
immutable = require 'immutable'
|
||||
{:insert, :remove, :concat} = table
|
||||
{:Lua, :Location} = require "lua_obj"
|
||||
|
||||
|
||||
common_methods = {
|
||||
__tostring: =>
|
||||
"#{@name}(#{repr(@value)})"
|
||||
with_value: (value)=> getmetatable(self)(value, @source)
|
||||
}
|
||||
fields = {"value","source"}
|
||||
|
||||
Types = {}
|
||||
Types.DictEntry = immutable({"key","value"}, {name:"DictEntry"})
|
||||
Types.is_node = (n)->
|
||||
type(n) == 'userdata' and getmetatable(n) and Types[n.type] == getmetatable(n)
|
||||
|
||||
-- Helper method:
|
||||
Tree = (name, methods)->
|
||||
with methods
|
||||
.__tostring = => "#{@name}(#{repr(@value)})"
|
||||
.with_value = (value)=> getmetatable(self)(value, @source)
|
||||
.type = name
|
||||
.name = name
|
||||
|
||||
Types[name] = immutable {"value","source"}, methods
|
||||
|
||||
|
||||
Tree "File",
|
||||
as_lua: (nomsu)=>
|
||||
if #@value == 1
|
||||
return @value[1]\as_lua(nomsu)
|
||||
declared_locals = {}
|
||||
lua = Lua(@source)
|
||||
for i, line in ipairs @value
|
||||
line_lua = line\as_lua(nomsu)
|
||||
if not line_lua
|
||||
error("No lua produced by #{repr line}", 0)
|
||||
if i < #@value
|
||||
lua\append "\n"
|
||||
lua\convert_to_statements!
|
||||
lua\append line_lua
|
||||
lua\declare_locals!
|
||||
return lua
|
||||
|
||||
Tree "Nomsu",
|
||||
as_lua: (nomsu)=>
|
||||
Lua.Value(@source, "nomsu:parse(",repr(@source\get_text!),", ",repr(@source.filename),")")
|
||||
|
||||
Tree "Block",
|
||||
as_lua: (nomsu)=>
|
||||
if #@value == 1
|
||||
return @value[1]\as_lua(nomsu)
|
||||
lua = Lua(@source)
|
||||
for i,line in ipairs @value
|
||||
line_lua = line\as_lua(nomsu)
|
||||
if i < #@value
|
||||
lua\append "\n"
|
||||
line_lua\convert_to_statements!
|
||||
lua\append line_lua
|
||||
return lua
|
||||
|
||||
math_patt = re.compile [[ "%" (" " [*/^+-] " %")+ ]]
|
||||
Tree "Action",
|
||||
as_lua: (nomsu)=>
|
||||
stub = @get_stub!
|
||||
action = rawget(nomsu.environment.ACTIONS, stub)
|
||||
metadata = nomsu.action_metadata[action]
|
||||
if metadata and metadata.compile_time
|
||||
args = [arg for arg in *@value when arg.type != "Word"]
|
||||
-- Force all compile-time actions to take a tree location
|
||||
if metadata.arg_orders
|
||||
new_args = [args[p-1] for p in *metadata.arg_orders[stub]]
|
||||
args = new_args
|
||||
return action(Lua(@source), unpack(args))
|
||||
|
||||
lua = Lua.Value(@source)
|
||||
if not metadata and math_patt\match(stub)
|
||||
-- This is a bit of a hack, but this code handles arbitrarily complex
|
||||
-- math expressions like 2*x + 3^2 without having to define a single
|
||||
-- action for every possibility.
|
||||
for i,tok in ipairs @value
|
||||
if tok.type == "Word"
|
||||
lua\append tok.value
|
||||
else
|
||||
tok_lua = tok\as_lua(nomsu)
|
||||
unless tok_lua.is_value
|
||||
src = tok.source\get_text!
|
||||
error("non-expression value inside math expression: #{colored.yellow src}")
|
||||
lua\append tok_lua
|
||||
if i < #@value
|
||||
lua\append " "
|
||||
lua\parenthesize!
|
||||
return lua
|
||||
|
||||
args = {}
|
||||
for i, tok in ipairs @value
|
||||
if tok.type == "Word" then continue
|
||||
arg_lua = tok\as_lua(nomsu)
|
||||
unless arg_lua.is_value
|
||||
line, src = tok.source\get_line!, tok.source\get_text!
|
||||
error "#{line}: Cannot use:\n#{colored.yellow src}\nas an argument to #{stub}, since it's not an expression, it produces: #{repr arg_lua}", 0
|
||||
insert args, arg_lua
|
||||
|
||||
if metadata and metadata.arg_orders
|
||||
args = [args[p] for p in *metadata.arg_orders[stub]]
|
||||
|
||||
-- Not really worth bothering with ACTIONS.foo(...) style since almost every action
|
||||
-- has arguments, so it won't work
|
||||
lua\append "ACTIONS[",repr(stub),"]("
|
||||
for i, arg in ipairs args
|
||||
lua\append arg
|
||||
if i < #args then lua\append ", "
|
||||
lua\append ")"
|
||||
return lua
|
||||
|
||||
get_stub: (include_names=false)=>
|
||||
bits = if include_names
|
||||
[(t.type == "Word" and t.value or "%#{t.value}") for t in *@value]
|
||||
else [(t.type == "Word" and t.value or "%") for t in *@value]
|
||||
return concat(bits, " ")
|
||||
|
||||
|
||||
Tree "Text",
|
||||
as_lua: (nomsu)=>
|
||||
lua = Lua.Value(@source)
|
||||
string_buffer = ""
|
||||
for bit in *@value
|
||||
if type(bit) == "string"
|
||||
string_buffer ..= bit
|
||||
continue
|
||||
if string_buffer ~= ""
|
||||
if #lua.bits > 0 then lua\append ".."
|
||||
lua\append repr(string_buffer)
|
||||
string_buffer = ""
|
||||
bit_lua = bit\as_lua(nomsu)
|
||||
unless bit_lua.is_value
|
||||
line, src = bit.source\get_line!, bit.source\get_text!
|
||||
error "#{line}: Cannot use #{colored.yellow bit} as a string interpolation value, since it's not an expression.", 0
|
||||
if #lua.bits > 0 then lua\append ".."
|
||||
if bit.type != "Text"
|
||||
bit_lua = Lua.Value(bit.source, "stringify(",bit_lua,")")
|
||||
lua\append bit_lua
|
||||
|
||||
if string_buffer ~= "" or #lua.bits == 0
|
||||
if #lua.bits > 0 then lua\append ".."
|
||||
lua\append repr(string_buffer)
|
||||
|
||||
if #lua.bits > 1
|
||||
lua\parenthesize!
|
||||
return lua
|
||||
|
||||
Tree "List",
|
||||
as_lua: (nomsu)=>
|
||||
lua = Lua.Value @source, "{"
|
||||
line_length = 0
|
||||
for i, item in ipairs @value
|
||||
item_lua = item\as_lua(nomsu)
|
||||
unless item_lua.is_value
|
||||
line, src = item.source\get_line!, item.source\get_text!
|
||||
error "#{line}: Cannot use #{colored.yellow src} as a list item, since it's not an expression.", 0
|
||||
lua\append item_lua
|
||||
newlines, last_line = tostring(item_lua)\match("^(.-)([^\n]*)$")
|
||||
if #newlines > 0
|
||||
line_length = #last_line
|
||||
else
|
||||
line_length += #last_line
|
||||
if i < #@value
|
||||
if line_length >= 80
|
||||
lua\append ",\n"
|
||||
line_length = 0
|
||||
else
|
||||
lua\append ", "
|
||||
line_length += 2
|
||||
lua\append "}"
|
||||
return lua
|
||||
|
||||
Tree "Dict",
|
||||
as_lua: (nomsu)=>
|
||||
lua = Lua.Value @source, "{"
|
||||
line_length = 0
|
||||
for i, entry in ipairs @value
|
||||
key_lua = entry.key\as_lua(nomsu)
|
||||
unless key_lua.is_value
|
||||
line, src = key.source\get_line!, key.source\get_text!
|
||||
error "#{line}: Cannot use #{colored.yellow src} as a dict key, since it's not an expression.", 0
|
||||
value_lua = entry.value\as_lua(nomsu)
|
||||
unless value_lua.is_value
|
||||
line, src = value.source\get_line!, value.source\get_text!
|
||||
error "#{line}: Cannot use #{colored.yellow src} as a dict value, since it's not an expression.", 0
|
||||
key_str = tostring(key_lua)\match([=[["']([a-zA-Z_][a-zA-Z0-9_]*)['"]]=])
|
||||
if key_str
|
||||
lua\append key_str,"=",value_lua
|
||||
elseif tostring(key_lua)\sub(1,1) == "["
|
||||
-- NOTE: this *must* use a space after the [ to avoid freaking out
|
||||
-- Lua's parser if the inner expression is a long string. Lua
|
||||
-- parses x[[[y]]] as x("[y]"), not as x["y"]
|
||||
lua\append "[ ",key_lua,"]=",value_lua
|
||||
else
|
||||
lua\append "[",key_lua,"]=",value_lua
|
||||
|
||||
-- TODO: maybe make this more accurate? It's only a heuristic, so eh...
|
||||
newlines, last_line = ("[#{key_lua}=#{value_lua}")\match("^(.-)([^\n]*)$")
|
||||
if #newlines > 0
|
||||
line_length = #last_line
|
||||
else
|
||||
line_length += #last_line
|
||||
if i < #@value
|
||||
if line_length >= 80
|
||||
lua\append ",\n"
|
||||
line_length = 0
|
||||
else
|
||||
lua\append ", "
|
||||
line_length += 2
|
||||
lua\append "}"
|
||||
return lua
|
||||
|
||||
Tree "IndexChain",
|
||||
as_lua: (nomsu)=>
|
||||
lua = @value[1]\as_lua(nomsu)
|
||||
unless lua.is_value
|
||||
line, src = @value[1].source\get_line!, @value[1].source\get_text!
|
||||
error "#{line}: Cannot index #{colored.yellow src}, since it's not an expression.", 0
|
||||
last_char = tostring(lua)\sub(-1,-1)
|
||||
if last_char == "}" or last_char == '"' or last_char == "]"
|
||||
lua\parenthesize!
|
||||
|
||||
for i=2,#@value
|
||||
key = @value[i]
|
||||
if key.type == 'Text' and #key.value == 1 and type(key.value[1]) == 'string' and key.value[1]\match("^[a-zA-Z_][a-zA-Z0-9_]$")
|
||||
lua\append ".#{key.value[1]}"
|
||||
continue
|
||||
key_lua = key\as_lua(nomsu)
|
||||
unless key_lua.is_value
|
||||
line, src = key.source\get_line!, key.source\get_text!
|
||||
error "#{line}: Cannot use #{colored.yellow src} as an index, since it's not an expression.", 0
|
||||
-- NOTE: this *must* use a space after the [ to avoid freaking out
|
||||
-- Lua's parser if the inner expression is a long string. Lua
|
||||
-- parses x[[[y]]] as x("[y]"), not as x["y"]
|
||||
if tostring(key_lua)\sub(1,1) == '['
|
||||
lua\append "[ ",key_lua,"]"
|
||||
else
|
||||
lua\append "[",key_lua,"]"
|
||||
return lua
|
||||
|
||||
Tree "Number",
|
||||
as_lua: (nomsu)=>
|
||||
Lua.Value(@source, tostring(@value))
|
||||
|
||||
Tree "Var",
|
||||
as_lua: (nomsu)=>
|
||||
lua_id = "_"..(@value\gsub "%W", (verboten)->
|
||||
if verboten == "_" then "__" else ("_%x")\format(verboten\byte!))
|
||||
Lua.Value(@source, lua_id)
|
||||
|
||||
Tree "Word",
|
||||
as_lua: (nomsu)=>
|
||||
error("Attempt to convert Word to lua")
|
||||
|
||||
Tree "Comment",
|
||||
as_lua: (nomsu)=>
|
||||
Lua(@source, "--"..@value\gsub("\n","\n--").."\n")
|
||||
|
||||
return Types
|
Loading…
Reference in New Issue
Block a user