Initial working version.
This commit is contained in:
parent
af9dc07025
commit
6f6c4377b2
@ -144,7 +144,7 @@ immediately
|
||||
for %var in %start to %stop via %step %body
|
||||
..to
|
||||
# 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)"
|
||||
assume (%var.type is "Var") or barf "Loop expected variable, not: \%var"
|
||||
%lua <-
|
||||
Lua ".."
|
||||
for \(%var as lua expr)=\(%start as lua expr),\(%stop as lua expr),\(%step as lua expr) do
|
||||
@ -153,7 +153,7 @@ immediately
|
||||
%body has subtree % where
|
||||
(%.type = "Action") and
|
||||
((%'s stub) is "do next %") and
|
||||
%.value.3.value is %var.value
|
||||
%.3 = %var
|
||||
..: to %lua write "\n ::continue_\(%var as lua identifier)::;"
|
||||
to %lua write "\nend --numeric for-loop"
|
||||
|
||||
@ -161,7 +161,7 @@ immediately
|
||||
%body has subtree % where
|
||||
(%.type = "Action") and
|
||||
((%'s stub) is "stop %") and
|
||||
%.value.2.value is %var.value
|
||||
%.2 = %var
|
||||
..
|
||||
%lua <-
|
||||
Lua ".."
|
||||
@ -179,7 +179,7 @@ immediately
|
||||
immediately
|
||||
compile [for %var in %iterable %body] to
|
||||
# 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)"
|
||||
assume (%var.type is "Var") or barf "Loop expected variable, not: \%var"
|
||||
%lua <-
|
||||
Lua ".."
|
||||
for i,\(%var as lua identifier) in ipairs(\(%iterable as lua expr)) do
|
||||
@ -188,14 +188,14 @@ immediately
|
||||
%body has subtree % where
|
||||
(%.type = "Action") and
|
||||
((%'s stub) is "do next %") and
|
||||
%.value.3.value is %var.value
|
||||
%.3 = %var
|
||||
..: to %lua write (Lua "\n ::continue_\(%var as lua identifier)::;")
|
||||
to %lua write "\nend --foreach-loop"
|
||||
if
|
||||
%body has subtree % where
|
||||
(%.type = "Action") and
|
||||
((%'s stub) is "stop %") and
|
||||
%.value.2.value is %var.value
|
||||
%.2 = %var
|
||||
..
|
||||
%lua <-
|
||||
Lua ".."
|
||||
@ -209,8 +209,8 @@ immediately
|
||||
immediately
|
||||
compile [for %key = %value in %iterable %body] to
|
||||
# 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)"
|
||||
assume (%key.type is "Var") or barf "Loop expected variable, not: \%key"
|
||||
assume (%value.type is "Var") or barf "Loop expected variable, not: \%value"
|
||||
%lua <-
|
||||
Lua ".."
|
||||
for \(%key as lua identifier),\(%value as lua identifier) in pairs(\(%iterable as lua expr)) do
|
||||
@ -219,14 +219,14 @@ immediately
|
||||
%body has subtree % where
|
||||
(%.type = "Action") and
|
||||
((%'s stub) is "do next %") and
|
||||
%.value.3.value is %key.value
|
||||
%.3 = %key
|
||||
..: to %lua write (Lua "\n ::continue_\(%key as lua identifier)::;")
|
||||
|
||||
if
|
||||
%body has subtree % where
|
||||
(%.type = "Action") and
|
||||
((%'s stub) is "do next %") and
|
||||
%.value.3.value is %value.value
|
||||
%.3 = %value
|
||||
..: to %lua write (Lua "\n ::continue_\(%value as lua identifier)::;")
|
||||
to %lua write "\nend --foreach-loop"
|
||||
|
||||
@ -235,14 +235,14 @@ immediately
|
||||
%body has subtree % where
|
||||
(%.type = "Action") and
|
||||
((%'s stub) is "stop %") and
|
||||
%.value.2.value is %key.value
|
||||
%.2 = %key
|
||||
..: to %stop_labels write "\n::stop_\(%key as lua identifier)::;"
|
||||
|
||||
if
|
||||
%body has subtree % where
|
||||
(%.type = "Action") and
|
||||
((%'s stub) is "stop %") and
|
||||
%.value.2.value is %value.value
|
||||
%.2 = %value
|
||||
..: to %stop_labels write "\n::stop_\(%value as lua identifier)::;"
|
||||
|
||||
if: (length of %stop_labels) > 0
|
||||
@ -262,15 +262,14 @@ immediately
|
||||
%is_first <- (yes)
|
||||
%seen_else <- (no)
|
||||
%branches <-
|
||||
%body.value if (%body.type = "Block") else [%body]
|
||||
%body if (%body.type = "Block") else [%body]
|
||||
for %func_call in %branches
|
||||
assume (%func_call.type is "Action") or barf ".."
|
||||
Invalid format for 'when' statement. Only '*' blocks are allowed.
|
||||
%tokens <- %func_call.value
|
||||
with {..}
|
||||
%star: %tokens.1
|
||||
%condition: %tokens.2
|
||||
%action: %tokens.3
|
||||
%star: %func_call.1
|
||||
%condition: %func_call.2
|
||||
%action: %func_call.3
|
||||
..
|
||||
assume ((%star and (%star.type is "Word")) and (%star.value is "*")) or barf ".."
|
||||
Invalid format for 'when' statement. Lines must begin with '*'
|
||||
@ -311,12 +310,11 @@ immediately
|
||||
%is_first <- (yes)
|
||||
%seen_else <- (no)
|
||||
%branches <-
|
||||
%body.value if (%body.type = "Block") else [%body]
|
||||
%body if (%body.type = "Block") else [%body]
|
||||
for %func_call in %branches
|
||||
assume (%func_call.type is "Action") or barf ".."
|
||||
Invalid format for 'when' statement. Only '*' blocks are allowed.
|
||||
%tokens <- %func_call.value
|
||||
with {%star:%tokens.1, %condition:%tokens.2, %action:%tokens.3}
|
||||
with {%star:%func_call.1, %condition:%func_call.2, %action:%func_call.3}
|
||||
assume ((%star and (%star.type is "Word")) and (%star.value is "*")) or barf ".."
|
||||
Invalid format for 'when' statement. Lines must begin with '*'
|
||||
assume %condition or barf ".."
|
||||
|
@ -42,7 +42,7 @@ compile [all of %items, all %items] to
|
||||
unless: (%items' "type") is "List"
|
||||
return: Lua value "utils.all(\(%items as lua expr))"
|
||||
%clauses <- []
|
||||
for % in (%items' "value")
|
||||
for % in %items
|
||||
lua> "table.insert(\%clauses, \(% as lua expr));"
|
||||
return: Lua value "(\(%clauses joined with " and "))"
|
||||
parse [not all of %items, not all %items] as: not (all of %items)
|
||||
@ -50,7 +50,7 @@ compile [any of %items, any %items] to
|
||||
unless: (%items' "type") is "List"
|
||||
return: Lua value "utils.any(\(%items as lua expr))"
|
||||
%clauses <- []
|
||||
for % in (%items' "value")
|
||||
for % in %items
|
||||
lua> "table.insert(\%clauses, \(% as lua expr));"
|
||||
return: Lua value "(\(%clauses joined with " or "))"
|
||||
parse [none of %items, none %items] as: not (any of %items)
|
||||
@ -58,14 +58,14 @@ compile [sum of %items, sum %items] to
|
||||
unless: (%items' "type") is "List"
|
||||
return: Lua value "utils.sum(\(%items as lua expr))"
|
||||
%clauses <- []
|
||||
for % in (%items' "value")
|
||||
for % in %items
|
||||
lua> "table.insert(\%clauses, \(% as lua expr));"
|
||||
return: Lua value "(\(%clauses joined with " + "))"
|
||||
compile [product of %items, product %items] to
|
||||
unless: (%items' "type") is "List"
|
||||
return: Lua value "utils.product(\(%items as lua expr))"
|
||||
%clauses <- []
|
||||
for % in (%items' "value")
|
||||
for % in %items
|
||||
lua> "table.insert(\%clauses, \(% as lua expr));"
|
||||
return: Lua value "(\(%clauses joined with " * "))"
|
||||
action [avg of %items, average of %items]
|
||||
|
@ -6,9 +6,9 @@
|
||||
immediately
|
||||
lua> ".."
|
||||
nomsu:define_compile_action("compile %actions to %lua", function(tree, \%actions, \%lua)
|
||||
local lua = Lua(tree.source, "nomsu:define_compile_action(")
|
||||
local lua = Lua(nil, "nomsu:define_compile_action(")
|
||||
local stubs = {}
|
||||
for i, action in ipairs(\%actions.value) do
|
||||
for i, action in ipairs(\%actions) do
|
||||
stubs[i] = action:get_stub(true)
|
||||
end
|
||||
stubs = repr(stubs)
|
||||
@ -19,7 +19,7 @@ immediately
|
||||
end
|
||||
lua:append("function(tree")
|
||||
local args = {}
|
||||
for i,tok in ipairs(\%actions.value[1].value) do
|
||||
for i,tok in ipairs(\%actions[1]) do
|
||||
if tok.type == "Var" then args[#args+1] = tok:as_lua(nomsu) end
|
||||
end
|
||||
for i, arg in ipairs(args) do
|
||||
@ -36,9 +36,9 @@ immediately
|
||||
immediately
|
||||
compile [action %actions %body] to
|
||||
lua> ".."
|
||||
local lua = Lua(tree.source, "nomsu:define_action(")
|
||||
local lua = Lua(nil, "nomsu:define_action(")
|
||||
local stubs = {}
|
||||
for i, action in ipairs(\%actions.value) do
|
||||
for i, action in ipairs(\%actions) do
|
||||
stubs[i] = action:get_stub(true)
|
||||
end
|
||||
stubs = repr(stubs)
|
||||
@ -49,7 +49,7 @@ immediately
|
||||
end
|
||||
lua:append("function(")
|
||||
local args = {}
|
||||
for i,tok in ipairs(\%actions.value[1].value) do
|
||||
for i,tok in ipairs(\%actions[1]) do
|
||||
if tok.type == "Var" then args[#args+1] = tok:as_lua(nomsu) end
|
||||
end
|
||||
for i, arg in ipairs(args) do
|
||||
@ -66,9 +66,9 @@ immediately
|
||||
immediately
|
||||
compile [parse %shorthand as %longhand] to
|
||||
lua> ".."
|
||||
local lua = Lua(tree.source, "nomsu:define_compile_action(")
|
||||
local lua = Lua(nil, "nomsu:define_compile_action(")
|
||||
local stubs = {}
|
||||
for i, action in ipairs(\%shorthand.value) do
|
||||
for i, action in ipairs(\%shorthand) do
|
||||
stubs[i] = action:get_stub(true)
|
||||
end
|
||||
stubs = repr(stubs)
|
||||
@ -79,24 +79,24 @@ immediately
|
||||
end
|
||||
lua:append("function(tree")
|
||||
local replacements = {}
|
||||
for i,tok in ipairs(\%shorthand.value[1].value) do
|
||||
for i,tok in ipairs(\%shorthand[1]) do
|
||||
if tok.type == "Var" then
|
||||
local lua_var = tostring(tok:as_lua(nomsu))
|
||||
replacements[tok.value] = lua_var
|
||||
replacements[tok] = lua_var
|
||||
lua:append(", ", lua_var)
|
||||
end
|
||||
end
|
||||
local function make_tree(t)
|
||||
if Tuple:is_instance(t) then
|
||||
if replacements[t] then
|
||||
return replacements[t]
|
||||
elseif type(t) ~= 'table' and type(t) ~= 'userdata' then
|
||||
return repr(t)
|
||||
elseif t.is_multi then
|
||||
local bits = {}
|
||||
for i, entry in ipairs(t) do
|
||||
bits[i] = make_tree(entry)
|
||||
end
|
||||
return "Tuple("..table.concat(bits, ", ")..")"
|
||||
elseif type(t) ~= 'table' and type(t) ~= 'userdata' then
|
||||
return repr(t)
|
||||
elseif t.type == "Var" and replacements[t.value] then
|
||||
return replacements[t.value]
|
||||
return t.type.."("..table.concat(bits, ", ")..")"
|
||||
else
|
||||
return t.type.."("..make_tree(t.value)..")"
|
||||
end
|
||||
@ -118,6 +118,12 @@ action [remove action %stub]
|
||||
ARG_ORDERS[fn] = nil
|
||||
|
||||
immediately
|
||||
action [%tree as nomsu]
|
||||
=lua "\%tree:as_nomsu()"
|
||||
|
||||
action [%tree as inline nomsu]
|
||||
=lua "\%tree:as_nomsu(true)"
|
||||
|
||||
action [%tree as lua]
|
||||
=lua "\%tree:as_lua(nomsu)"
|
||||
|
||||
@ -125,7 +131,7 @@ immediately
|
||||
lua> ".."
|
||||
local lua = \%tree:as_lua(nomsu)
|
||||
if not lua.is_value then
|
||||
error("Invalid thing to convert to lua expr: "..\%tree.source:get_text())
|
||||
error("Invalid thing to convert to lua expr: "..\%tree)
|
||||
end
|
||||
return lua
|
||||
|
||||
@ -156,9 +162,6 @@ immediately
|
||||
parse [to %var write %code] as: lua> "\%var:append(\%code);"
|
||||
|
||||
immediately
|
||||
action [%tree's source code, %tree' source code]
|
||||
=lua "\%tree.source:get_text()"
|
||||
|
||||
compile [repr %obj] to: Lua value "repr(\(%obj as lua expr))"
|
||||
compile [%obj as text] to: Lua value "tostring(\(%obj as lua expr))"
|
||||
compile [type of %obj] to: Lua value "type(\(%obj as lua expr))"
|
||||
@ -170,31 +173,23 @@ immediately
|
||||
# Compiler tools
|
||||
immediately
|
||||
compile [run %code] to
|
||||
Lua value "nomsu:run(Nomsu(\(=lua "tostring(tree.source)"), \(%code as lua expr)))"
|
||||
Lua value "nomsu:run(Nomsu(nil, \(%code as lua expr)))"
|
||||
|
||||
immediately
|
||||
compile [show lua %block] to
|
||||
lua> ".."
|
||||
local \%lua = \%block:as_lua(nomsu);
|
||||
return Lua(\%block.source, "print(", repr(tostring(\%lua)), ");");
|
||||
return Lua(nil, "print(", repr(tostring(\%lua)), ");");
|
||||
|
||||
immediately
|
||||
compile [say %message] to
|
||||
lua> ".."
|
||||
if \%message.type == "Text" then
|
||||
return Lua(tree.source, "print(", \(%message as lua expr), ");");
|
||||
return Lua(nil, "print(", \(%message as lua expr), ");");
|
||||
else
|
||||
return Lua(tree.source, "print(stringify(", \(%message as lua expr), "));");
|
||||
return Lua(nil, "print(stringify(", \(%message as lua expr), "));");
|
||||
end
|
||||
|
||||
immediately
|
||||
compile [source] to: Lua value "tree.source"
|
||||
|
||||
#
|
||||
immediately
|
||||
action [Lua %]: Lua (=lua "tree.source") %
|
||||
action [Lua value %]: Lua value (=lua "tree.source") %
|
||||
|
||||
# Return
|
||||
immediately
|
||||
# Return statement is wrapped in a do..end block because Lua is unhappy if you
|
||||
@ -207,7 +202,7 @@ immediately
|
||||
compile [barf] to: Lua "error(nil, 0);"
|
||||
compile [barf %msg] to: Lua "error(\(%msg as lua expr), 0);"
|
||||
compile [assume %condition] to
|
||||
lua> "local \%assumption = 'Assumption failed: '..tostring(\%condition.source:get_text());"
|
||||
lua> "local \%assumption = 'Assumption failed: '..tostring(\%condition:as_nomsu());"
|
||||
return
|
||||
Lua ".."
|
||||
if not \(%condition as lua expr) then
|
||||
|
@ -28,18 +28,18 @@ immediately
|
||||
local safe = {Text=true, Number=true};
|
||||
local a_lua, b_lua = \%a:as_lua(nomsu), \%b:as_lua(nomsu);
|
||||
if safe[\%a.type] or safe[\%b.type] then
|
||||
return Lua.Value(tree.source, "(", a_lua, " == ", b_lua, ")");
|
||||
return Lua.Value(nil, "(", a_lua, " == ", b_lua, ")");
|
||||
else
|
||||
return Lua.Value(tree.source, "utils.equivalent(", a_lua, ", ", b_lua, ")");
|
||||
return Lua.Value(nil, "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 = \%a:as_lua(nomsu), \%b:as_lua(nomsu);
|
||||
if safe[\%a.type] or safe[\%b.type] then
|
||||
return Lua.Value(tree.source, "(", a_lua, " ~= ", b_lua, ")");
|
||||
return Lua.Value(nil, "(", a_lua, " ~= ", b_lua, ")");
|
||||
else
|
||||
return Lua.Value(tree.source, "(not utils.equivalent(", a_lua, ", ", b_lua, "))");
|
||||
return Lua.Value(nil, "(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: Lua value "nomsu.ids[\(% as lua expr)]"
|
||||
@ -48,11 +48,11 @@ immediately
|
||||
immediately
|
||||
compile [%var <- %value] to
|
||||
lua> "local \%var_lua = \%var:as_lua(nomsu);"
|
||||
assume %var_lua.is_value or barf "Invalid target for assignment: \(%var's source code)"
|
||||
assume %var_lua.is_value or barf "Invalid target for assignment: \%var"
|
||||
lua> "local \%value_lua = \%value:as_lua(nomsu);"
|
||||
assume %value_lua.is_value or barf "Invalid value for assignment: \(%value's source code)"
|
||||
assume %value_lua.is_value or barf "Invalid value for assignment: \%value"
|
||||
lua> ".."
|
||||
local lua = Lua(tree.source, \%var_lua, ' = ', \%value_lua, ';');
|
||||
local lua = Lua(nil, \%var_lua, ' = ', \%value_lua, ';');
|
||||
if \%var.type == 'Var' then
|
||||
lua:add_free_vars({\%var});
|
||||
end
|
||||
@ -62,11 +62,11 @@ immediately
|
||||
# Simultaneous mutli-assignments like: x,y,z = 1,x,3;
|
||||
compile [<- %assignments] to
|
||||
assume ((%assignments' "type") is "Dict") or barf ".."
|
||||
Expected a Dict for the assignments part of '<- %' statement, not \(%assignments' source code)
|
||||
Expected a Dict for the assignments part of '<- %' statement, not \%assignments
|
||||
lua> ".."
|
||||
local lhs, rhs = Lua(tree.source), Lua(\%assignments.source);
|
||||
for i, item in ipairs(\%assignments.value) do
|
||||
local target, value = item.value[1], item.value[2];
|
||||
local lhs, rhs = Lua(), Lua();
|
||||
for i, item in ipairs(\%assignments) do
|
||||
local target, value = item[1], item[2];
|
||||
local target_lua = target:as_lua(nomsu);
|
||||
if not target_lua.is_value then error("Invalid target for assignment: "..target:get_src()); end
|
||||
local value_lua = value:as_lua(nomsu);
|
||||
@ -81,35 +81,35 @@ immediately
|
||||
lhs:append(target_lua);
|
||||
rhs:append(value_lua);
|
||||
end
|
||||
return Lua(tree.source, lhs, " = ", rhs, ";");
|
||||
return Lua(nil, lhs, " = ", rhs, ";");
|
||||
|
||||
immediately
|
||||
compile [external %var <- %value] to
|
||||
%var_lua <- (%var as lua)
|
||||
assume %var_lua.is_value or barf "Invalid target for assignment: \(%var's source code)"
|
||||
assume %var_lua.is_value or barf "Invalid target for assignment: \%var"
|
||||
%value_lua <- (%value as lua)
|
||||
assume %value_lua.is_value or barf "Invalid value for assignment: \(%value's source code)"
|
||||
assume %value_lua.is_value or barf "Invalid value for assignment: \%value"
|
||||
return: Lua "\(%var_lua) = \(%value_lua);"
|
||||
|
||||
compile [with external %externs %body] to
|
||||
%body_lua <- (%body as lua statements)
|
||||
lua> "\%body_lua:remove_free_vars(\(%externs.value));"
|
||||
lua> "\%body_lua:remove_free_vars(\(%externs));"
|
||||
return %body_lua
|
||||
|
||||
compile [with %assignments %body] to
|
||||
%lua <- (%body as lua statements)
|
||||
lua> ".."
|
||||
local lhs, rhs = Lua(tree.source), Lua(\%assignments.source);
|
||||
local lhs, rhs = Lua(), Lua();
|
||||
local vars = {};
|
||||
for i, item in ipairs(\%assignments.value) do
|
||||
local target, value = item.value[1], item.value[2];
|
||||
for i, item in ipairs(\%assignments) do
|
||||
local target, value = item[1], item[2];
|
||||
if not target.type == "Var" then
|
||||
error("Invalid target for 'with' assignment: "..tostring(target.source:get_text()));
|
||||
error("Invalid target for 'with' assignment: "..tostring(target));
|
||||
end
|
||||
local target_lua = target:as_lua(nomsu);
|
||||
local value_lua = value:as_lua(nomsu);
|
||||
if not value_lua.is_value then
|
||||
error("Invalid value for assignment: "..tostring(value.source:get_text()));
|
||||
error("Invalid value for assignment: "..tostring(value));
|
||||
end
|
||||
if target.type == "Var" then
|
||||
lhs:add_free_vars({target});
|
||||
|
@ -27,7 +27,7 @@ lua> ".."
|
||||
};
|
||||
for name, e in pairs(escapes) do
|
||||
local lua = "'"..e.."'";
|
||||
nomsu:define_compile_action(name, function(tree) return Lua.Value(tree.source, lua); end);
|
||||
nomsu:define_compile_action(name, function(tree) return Lua.Value(nil, lua); end);
|
||||
end
|
||||
local colors = {
|
||||
["reset color"]="\\\\27[0m", bright="\\\\27[1m", dim="\\\\27[2m", underscore="\\\\27[4m",
|
||||
@ -42,9 +42,9 @@ lua> ".."
|
||||
for name, c in pairs(colors) do
|
||||
local color = "'"..c.."'";
|
||||
local reset = "'"..colors["reset color"].."'";
|
||||
nomsu:define_compile_action(name, function(tree) return Lua.Value(tree.source, color); end);
|
||||
nomsu:define_compile_action(name, function(tree) return Lua.Value(nil, color); end);
|
||||
nomsu:define_compile_action(name.." %", function(\%)
|
||||
return Lua.Value(tree.source, color, "..", \%:as_lua(nomsu), "..", reset);
|
||||
return Lua.Value(nil, color, "..", \%:as_lua(nomsu), "..", reset);
|
||||
end);
|
||||
end
|
||||
end
|
||||
|
68
nomsu.lua
68
nomsu.lua
@ -146,9 +146,6 @@ local Types = require("nomsu_tree")
|
||||
local NOMSU_DEFS
|
||||
do
|
||||
local _with_0 = { }
|
||||
_with_0.Tuple = function(values)
|
||||
return Tuple(unpack(values))
|
||||
end
|
||||
_with_0.nl = P("\r") ^ -1 * P("\n")
|
||||
_with_0.ws = S(" \t")
|
||||
_with_0.tonumber = tonumber
|
||||
@ -228,16 +225,8 @@ end
|
||||
setmetatable(NOMSU_DEFS, {
|
||||
__index = function(self, key)
|
||||
local make_node
|
||||
make_node = function(start, value, stop)
|
||||
if type(value) == 'table' then
|
||||
error("Not a tuple: " .. tostring(repr(value)))
|
||||
end
|
||||
local source = lpeg.userdata.source_code.source
|
||||
start = start + (source.start - 1)
|
||||
stop = stop + (source.start - 1)
|
||||
source = Source(source.filename, start, stop - 1)
|
||||
local node = Types[key](value, source)
|
||||
return node
|
||||
make_node = function(src, ...)
|
||||
return Types[key](...)
|
||||
end
|
||||
self[key] = make_node
|
||||
return make_node
|
||||
@ -250,7 +239,7 @@ do
|
||||
anon_def <- ({ident} (" "*) ":"
|
||||
{((%nl " "+ [^%nl]*)+) / ([^%nl]*)}) -> "%1 <- %2"
|
||||
captured_def <- ({ident} (" "*) "(" {ident} ")" (" "*) ":"
|
||||
{((%nl " "+ [^%nl]*)+) / ([^%nl]*)}) -> "%1 <- ({} %3 {}) -> %2"
|
||||
{((%nl " "+ [^%nl]*)+) / ([^%nl]*)}) -> "%1 <- (({} %3) -> %2)"
|
||||
ident <- [a-zA-Z_][a-zA-Z0-9_]*
|
||||
comment <- "--" [^%nl]*
|
||||
]])
|
||||
@ -430,12 +419,6 @@ do
|
||||
run_lua = function(self, lua)
|
||||
assert(type(lua) ~= 'string', "Attempt to run lua string instead of Lua (object)")
|
||||
local lua_string = tostring(lua)
|
||||
if rawget(FILE_CACHE, lua.source.filename) == nil then
|
||||
FILE_CACHE[lua.source.filename] = lua_string
|
||||
end
|
||||
if rawget(FILE_CACHE, lua.source) == nil then
|
||||
FILE_CACHE[lua.source] = lua_string
|
||||
end
|
||||
local run_lua_fn, err = load(lua_string, filename, "t", self.environment)
|
||||
if not run_lua_fn then
|
||||
local n = 1
|
||||
@ -450,10 +433,10 @@ do
|
||||
return run_lua_fn()
|
||||
end,
|
||||
tree_to_value = function(self, tree)
|
||||
if tree.type == 'Text' and #tree.value == 1 and type(tree.value[1]) == 'string' then
|
||||
return tree.value[1]
|
||||
if tree.type == 'Text' and #tree == 1 and type(tree[1]) == 'string' then
|
||||
return tree[1]
|
||||
end
|
||||
local lua = Lua(tree.source, "return ", tree:as_lua(self), ";")
|
||||
local lua = Lua(nil, "return ", tree:as_lua(self), ";")
|
||||
return self:run_lua(lua)
|
||||
end,
|
||||
walk_tree = function(self, tree, depth)
|
||||
@ -464,10 +447,9 @@ do
|
||||
if not (Types.is_node(tree)) then
|
||||
return
|
||||
end
|
||||
if Tuple:is_instance(tree.value) then
|
||||
local _list_0 = tree.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local v = _list_0[_index_0]
|
||||
if tree.is_multi then
|
||||
for _index_0 = 1, #tree do
|
||||
local v = tree[_index_0]
|
||||
self:walk_tree(v, depth + 1)
|
||||
end
|
||||
else
|
||||
@ -502,7 +484,7 @@ do
|
||||
local lua = _block:as_lua(nomsu):as_statements()
|
||||
lua:declare_locals()
|
||||
nomsu:run_lua(lua)
|
||||
return Lua(self.source, "if IMMEDIATE then\n ", lua, "\nend")
|
||||
return Lua(nil, "if IMMEDIATE then\n ", lua, "\nend")
|
||||
end)
|
||||
local add_lua_string_bits
|
||||
add_lua_string_bits = function(lua, code)
|
||||
@ -510,46 +492,42 @@ do
|
||||
lua:append(", ", code:as_lua(nomsu))
|
||||
return
|
||||
end
|
||||
local _list_0 = code.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local bit = _list_0[_index_0]
|
||||
for _index_0 = 1, #code do
|
||||
local bit = code[_index_0]
|
||||
lua:append(", ")
|
||||
if type(bit) == "string" then
|
||||
lua:append(repr(bit))
|
||||
else
|
||||
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(src)) .. " as a string interpolation value, since it's not an expression.")
|
||||
error("Cannot use " .. tostring(colored.yellow(repr(bit))) .. " as a string interpolation value, since it's not an expression.")
|
||||
end
|
||||
lua:append(bit_lua)
|
||||
end
|
||||
end
|
||||
end
|
||||
self:define_compile_action("Lua %code", function(self, _code)
|
||||
local lua = Lua.Value(self.source, "Lua(", tostring(_code.source))
|
||||
local lua = Lua.Value(nil, "Lua(nil")
|
||||
add_lua_string_bits(lua, _code)
|
||||
lua:append(")")
|
||||
return lua
|
||||
end)
|
||||
self:define_compile_action("Lua value %code", function(self, _code)
|
||||
local lua = Lua.Value(self.source, "Lua.Value(", tostring(_code.source))
|
||||
local lua = Lua.Value(nil, "Lua.Value(nil")
|
||||
add_lua_string_bits(lua, _code)
|
||||
lua:append(")")
|
||||
return lua
|
||||
end)
|
||||
local add_lua_bits
|
||||
add_lua_bits = function(lua, code)
|
||||
local _list_0 = code.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local bit = _list_0[_index_0]
|
||||
for _index_0 = 1, #code do
|
||||
local bit = code[_index_0]
|
||||
if type(bit) == "string" then
|
||||
lua:append(bit)
|
||||
else
|
||||
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(src)) .. " as a string interpolation value, since it's not an expression.", 0)
|
||||
error("Cannot use " .. tostring(colored.yellow(repr(bit))) .. " as a string interpolation value, since it's not an expression.", 0)
|
||||
end
|
||||
lua:append(bit_lua)
|
||||
end
|
||||
@ -558,20 +536,20 @@ do
|
||||
end
|
||||
self:define_compile_action("lua> %code", function(self, _code)
|
||||
if _code.type ~= "Text" then
|
||||
return Lua(self.source, "nomsu:run_lua(", _code:as_lua(nomsu), ");")
|
||||
return Lua(nil, "nomsu:run_lua(", _code:as_lua(nomsu), ");")
|
||||
end
|
||||
return add_lua_bits(Lua(_code.source), _code)
|
||||
return add_lua_bits(Lua(), _code)
|
||||
end)
|
||||
self:define_compile_action("=lua %code", function(self, _code)
|
||||
if _code.type ~= "Text" then
|
||||
return Lua.Value(self.source, "nomsu:run_lua(", _code:as_lua(nomsu), ":as_statements('return '))")
|
||||
return Lua.Value(nil, "nomsu:run_lua(", _code:as_lua(nomsu), ":as_statements('return '))")
|
||||
end
|
||||
return add_lua_bits(Lua.Value(_code.source), _code)
|
||||
return add_lua_bits(Lua.Value(), _code)
|
||||
end)
|
||||
return self:define_compile_action("use %path", function(self, _path)
|
||||
local path = nomsu:tree_to_value(_path)
|
||||
nomsu:run_file(path)
|
||||
return Lua(self.source, "nomsu:run_file(" .. tostring(repr(path)) .. ");")
|
||||
return Lua(nil, "nomsu:run_file(" .. tostring(repr(path)) .. ");")
|
||||
end)
|
||||
end
|
||||
}
|
||||
|
58
nomsu.moon
58
nomsu.moon
@ -129,8 +129,6 @@ Types = require "nomsu_tree"
|
||||
|
||||
NOMSU_DEFS = with {}
|
||||
-- Newline supports either windows-style CR+LF or unix-style LF
|
||||
.Tuple = (values)->
|
||||
return Tuple(unpack(values))
|
||||
.nl = P("\r")^-1 * P("\n")
|
||||
.ws = S(" \t")
|
||||
.tonumber = tonumber
|
||||
@ -196,15 +194,8 @@ NOMSU_DEFS = with {}
|
||||
return true
|
||||
|
||||
setmetatable(NOMSU_DEFS, {__index:(key)=>
|
||||
make_node = (start, value, stop)->
|
||||
if type(value) == 'table' then error("Not a tuple: #{repr value}")-- = Tuple(value)
|
||||
--source = lpeg.userdata.source_code.source\sub(start,stop-1)
|
||||
source = lpeg.userdata.source_code.source
|
||||
start += source.start-1
|
||||
stop += source.start-1
|
||||
source = Source(source.filename, start, stop-1)
|
||||
node = Types[key](value, source)
|
||||
return node
|
||||
make_node = (src, ...)->
|
||||
Types[key](...)
|
||||
self[key] = make_node
|
||||
return make_node
|
||||
})
|
||||
@ -218,7 +209,7 @@ NOMSU_PATTERN = do
|
||||
anon_def <- ({ident} (" "*) ":"
|
||||
{((%nl " "+ [^%nl]*)+) / ([^%nl]*)}) -> "%1 <- %2"
|
||||
captured_def <- ({ident} (" "*) "(" {ident} ")" (" "*) ":"
|
||||
{((%nl " "+ [^%nl]*)+) / ([^%nl]*)}) -> "%1 <- ({} %3 {}) -> %2"
|
||||
{((%nl " "+ [^%nl]*)+) / ([^%nl]*)}) -> "%1 <- (({} %3) -> %2)"
|
||||
ident <- [a-zA-Z_][a-zA-Z0-9_]*
|
||||
comment <- "--" [^%nl]*
|
||||
]]
|
||||
@ -401,13 +392,6 @@ class NomsuCompiler
|
||||
run_lua: (lua)=>
|
||||
assert(type(lua) != 'string', "Attempt to run lua string instead of Lua (object)")
|
||||
lua_string = tostring(lua)
|
||||
--metadata = lua\make_offset_table!
|
||||
--LUA_METADATA[metadata.lua_filename] = metadata
|
||||
if rawget(FILE_CACHE, lua.source.filename) == nil
|
||||
FILE_CACHE[lua.source.filename] = lua_string
|
||||
if rawget(FILE_CACHE, lua.source) == nil
|
||||
FILE_CACHE[lua.source] = lua_string
|
||||
|
||||
run_lua_fn, err = load(lua_string, filename, "t", @environment)
|
||||
if not run_lua_fn
|
||||
n = 1
|
||||
@ -420,16 +404,16 @@ class NomsuCompiler
|
||||
|
||||
tree_to_value: (tree)=>
|
||||
-- Special case for text literals
|
||||
if tree.type == 'Text' and #tree.value == 1 and type(tree.value[1]) == 'string'
|
||||
return tree.value[1]
|
||||
lua = Lua(tree.source, "return ",tree\as_lua(@),";")
|
||||
if tree.type == 'Text' and #tree == 1 and type(tree[1]) == 'string'
|
||||
return tree[1]
|
||||
lua = Lua(nil, "return ",tree\as_lua(@),";")
|
||||
return @run_lua(lua)
|
||||
|
||||
walk_tree: (tree, depth=0)=>
|
||||
coroutine.yield(tree, depth)
|
||||
return unless Types.is_node(tree)
|
||||
if Tuple\is_instance(tree.value)
|
||||
for v in *tree.value
|
||||
if tree.is_multi
|
||||
for v in *tree
|
||||
@walk_tree(v, depth+1)
|
||||
else
|
||||
@walk_tree(v, depth+1)
|
||||
@ -451,61 +435,59 @@ class NomsuCompiler
|
||||
lua = _block\as_lua(nomsu)\as_statements!
|
||||
lua\declare_locals!
|
||||
nomsu\run_lua(lua)
|
||||
return Lua(@source, "if IMMEDIATE then\n ", lua, "\nend")
|
||||
return Lua(nil, "if IMMEDIATE then\n ", lua, "\nend")
|
||||
|
||||
add_lua_string_bits = (lua, code)->
|
||||
if code.type != "Text"
|
||||
lua\append ", ", code\as_lua(nomsu)
|
||||
return
|
||||
for bit in *code.value
|
||||
for bit in *code
|
||||
lua\append ", "
|
||||
if type(bit) == "string"
|
||||
lua\append repr(bit)
|
||||
else
|
||||
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 src} as a string interpolation value, since it's not an expression."
|
||||
error "Cannot use #{colored.yellow repr(bit)} as a string interpolation value, since it's not an expression."
|
||||
lua\append bit_lua
|
||||
|
||||
@define_compile_action "Lua %code", (_code)=>
|
||||
lua = Lua.Value(@source, "Lua(", tostring(_code.source))
|
||||
lua = Lua.Value(nil, "Lua(nil")
|
||||
add_lua_string_bits(lua, _code)
|
||||
lua\append ")"
|
||||
return lua
|
||||
|
||||
@define_compile_action "Lua value %code", (_code)=>
|
||||
lua = Lua.Value(@source, "Lua.Value(", tostring(_code.source))
|
||||
lua = Lua.Value(nil, "Lua.Value(nil")
|
||||
add_lua_string_bits(lua, _code)
|
||||
lua\append ")"
|
||||
return lua
|
||||
|
||||
add_lua_bits = (lua, code)->
|
||||
for bit in *code.value
|
||||
for bit in *code
|
||||
if type(bit) == "string"
|
||||
lua\append bit
|
||||
else
|
||||
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 src} as a string interpolation value, since it's not an expression.", 0
|
||||
error "Cannot use #{colored.yellow repr(bit)} as a string interpolation value, since it's not an expression.", 0
|
||||
lua\append bit_lua
|
||||
return lua
|
||||
|
||||
@define_compile_action "lua> %code", (_code)=>
|
||||
if _code.type != "Text"
|
||||
return Lua @source, "nomsu:run_lua(", _code\as_lua(nomsu), ");"
|
||||
return add_lua_bits(Lua(_code.source), _code)
|
||||
return Lua nil, "nomsu:run_lua(", _code\as_lua(nomsu), ");"
|
||||
return add_lua_bits(Lua!, _code)
|
||||
|
||||
@define_compile_action "=lua %code", (_code)=>
|
||||
if _code.type != "Text"
|
||||
return Lua.Value @source, "nomsu:run_lua(", _code\as_lua(nomsu), ":as_statements('return '))"
|
||||
return add_lua_bits(Lua.Value(_code.source), _code)
|
||||
return Lua.Value nil, "nomsu:run_lua(", _code\as_lua(nomsu), ":as_statements('return '))"
|
||||
return add_lua_bits(Lua.Value!, _code)
|
||||
|
||||
@define_compile_action "use %path", (_path)=>
|
||||
path = nomsu\tree_to_value(_path)
|
||||
nomsu\run_file(path)
|
||||
return Lua(@source, "nomsu:run_file(#{repr path});")
|
||||
return Lua(nil, "nomsu:run_file(#{repr path});")
|
||||
|
||||
-- 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
|
||||
|
40
nomsu.peg
40
nomsu.peg
@ -11,18 +11,16 @@ statement: (action / expression) (eol / (({} ([^%nl]* -> "Error while parsing li
|
||||
inline_statement: inline_action / inline_expression
|
||||
|
||||
inline_block (Block):
|
||||
{| inline_statement (%ws* ";" %ws* inline_statement)+ |} -> Tuple
|
||||
inline_statement (%ws* ";" %ws* inline_statement)+
|
||||
block (Block):
|
||||
{| statement (nodent (statement / (({} ([^%nl]* -> "Error while parsing block line")) => error)))+ |} -> Tuple
|
||||
statement (nodent (statement / (({} ([^%nl]* -> "Error while parsing block line")) => error)))+
|
||||
|
||||
inline_nomsu (Nomsu): "\" noindex_inline_expression
|
||||
indented_nomsu (Nomsu):
|
||||
inline_nomsu (EscapedNomsu): "\" noindex_inline_expression
|
||||
indented_nomsu (EscapedNomsu):
|
||||
"\" (noindex_inline_expression / (":" %ws* (inline_block / inline_action / inline_expression) eol) / indented_expression)
|
||||
|
||||
index_chain (IndexChain):
|
||||
{|
|
||||
noindex_inline_expression ("." (text_word / noindex_inline_expression))+
|
||||
|} -> Tuple
|
||||
|
||||
noindex_inline_expression:
|
||||
number / variable / inline_text / inline_list / inline_dict / inline_nomsu
|
||||
@ -47,30 +45,30 @@ expression:
|
||||
|
||||
-- Function calls need at least one word in them
|
||||
inline_action (Action):
|
||||
{| (inline_expression %ws*)* word (%ws* (inline_expression / word))*
|
||||
(%ws* ":" %ws* (inline_block / inline_action / inline_expression))?|} -> Tuple
|
||||
(inline_expression %ws*)* word (%ws* (inline_expression / word))*
|
||||
(%ws* ":" %ws* (inline_block / inline_action / inline_expression))?
|
||||
action (Action):
|
||||
{| (expression (dotdot? %ws*))* word ((dotdot? %ws*) (expression / word))* |} -> Tuple
|
||||
(expression (dotdot? %ws*))* word ((dotdot? %ws*) (expression / word))*
|
||||
|
||||
word (Word): { %operator / (!number plain_word) }
|
||||
|
||||
text_word (Text): {| {%operator / (!number plain_word)} |} -> Tuple
|
||||
text_word (Text): {%operator / (!number plain_word)}
|
||||
|
||||
inline_text (Text):
|
||||
!('".."' eol)
|
||||
'"' ({|
|
||||
'"' (
|
||||
({~ (('\"' -> '"') / ('\\' -> '\') / %escaped_char / [^%nl\"])+ ~}
|
||||
/ inline_text_interpolation)*
|
||||
|} -> Tuple) ('"' / (({} ([^%nl]*->'Failed to find a closing " mark on the same line')) => error))
|
||||
) ('"' / (({} ([^%nl]*->'Failed to find a closing " mark on the same line')) => error))
|
||||
|
||||
-- Have to use "%indent" instead of "indent" etc. to avoid messing up text lines that start with "#"
|
||||
indented_text (Text):
|
||||
'".."' eol %nl ({|
|
||||
'".."' eol %nl (
|
||||
{~ (%nl*) (%indent -> "") ~}
|
||||
({~
|
||||
(("\\" -> "\") / (("\" nodent "..") -> "")/ (%nl+ {~ %nodent -> "" ~}) / [^%nl\] / (!text_interpolation "\"))+
|
||||
~} / text_interpolation)*
|
||||
|} -> Tuple) (((!.) &%dedent) / (&(%nl %dedent)) / (({} (non_dedent_error -> "Error while parsing Text")) => error))
|
||||
) (((!.) &%dedent) / (&(%nl %dedent)) / (({} (non_dedent_error -> "Error while parsing Text")) => error))
|
||||
inline_text_interpolation:
|
||||
"\" (
|
||||
variable / inline_list / inline_dict / inline_text
|
||||
@ -93,12 +91,11 @@ variable (Var): "%" { plain_word? }
|
||||
|
||||
inline_list (List):
|
||||
!('[..]')
|
||||
"[" %ws* ({| (inline_list_item (comma inline_list_item)* comma?)? |} -> Tuple) %ws*
|
||||
"[" %ws* (inline_list_item (comma inline_list_item)* comma?)? %ws*
|
||||
("]" / (({} ([^%nl]*->"Failed to find a closing ] on the same line")) => error))
|
||||
indented_list (List):
|
||||
"[..]" indent ({|
|
||||
"[..]" indent
|
||||
list_line (nodent list_line)*
|
||||
|} -> Tuple)
|
||||
(dedent / (({} (non_dedent_error -> "Error while parsing list")) => error))
|
||||
list_line:
|
||||
((action / expression) !comma)
|
||||
@ -107,21 +104,20 @@ inline_list_item: inline_block / inline_action / inline_expression
|
||||
|
||||
inline_dict (Dict):
|
||||
!('{..}')
|
||||
"{" %ws* ({| (inline_dict_entry (comma inline_dict_entry)*)? |} -> Tuple) %ws*
|
||||
"{" %ws* (inline_dict_entry (comma inline_dict_entry)*)? %ws*
|
||||
("}"
|
||||
/ (({} (%ws* comma? (!. / &%nl)->"Failed to find a closing } on the same line")) => error)
|
||||
/ (({} ([^%nl]*->"Error while parsing dictionary")) => error))
|
||||
indented_dict (Dict):
|
||||
"{..}" indent ({|
|
||||
"{..}" indent
|
||||
dict_line (nodent dict_line)*
|
||||
|} -> Tuple)
|
||||
(dedent / (({} (non_dedent_error -> "Error while parsing dict")) => error))
|
||||
dict_line:
|
||||
(dict_entry !comma) / (inline_dict_entry (comma dict_line?)?)
|
||||
dict_entry(DictEntry):
|
||||
{| dict_key %ws* ":" %ws* (action / expression) |} -> Tuple
|
||||
dict_key %ws* ":" %ws* (action / expression)
|
||||
inline_dict_entry(DictEntry):
|
||||
{| dict_key %ws* (":" %ws* (inline_block / inline_action / inline_expression)?) |} -> Tuple
|
||||
dict_key %ws* (":" %ws* (inline_block / inline_action / inline_expression)?)
|
||||
dict_key:
|
||||
text_word / inline_expression
|
||||
|
||||
|
333
nomsu_tree.lua
333
nomsu_tree.lua
@ -18,59 +18,92 @@ 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)
|
||||
Tree = function(name, kind, methods)
|
||||
assert((kind == 'single') or (kind == 'multi'))
|
||||
local is_multi = (kind == 'multi')
|
||||
do
|
||||
methods.__tostring = function(self)
|
||||
return tostring(self.name) .. "(" .. tostring(repr(self.value)) .. ", " .. tostring(repr(self.source)) .. ")"
|
||||
end
|
||||
methods.with_value = function(self, value)
|
||||
return getmetatable(self)(value, self.source)
|
||||
return getmetatable(self)(value)
|
||||
end
|
||||
methods.type = name
|
||||
methods.name = name
|
||||
methods.original_nomsu = function(self)
|
||||
local leading_space = 0
|
||||
local src_file = FILE_CACHE[self.source.filename]
|
||||
while src_file:sub(self.source.start - leading_space - 1, self.source.start - leading_space - 1) == " " do
|
||||
leading_space = leading_space + 1
|
||||
end
|
||||
if src_file:sub(self.source.start - leading_space - 1, self.source.start - leading_space - 1) ~= "\n" then
|
||||
leading_space = 0
|
||||
end
|
||||
local ret = tostring(self.source:get_text()):gsub("\n" .. ((" "):rep(leading_space)), "\n")
|
||||
return ret
|
||||
end
|
||||
methods.map = function(self, fn)
|
||||
do
|
||||
local mapped = fn(self)
|
||||
if mapped then
|
||||
return mapped
|
||||
end
|
||||
end
|
||||
if Tuple:is_instance(self.value) then
|
||||
return self:with_value(Tuple(unpack((function()
|
||||
methods.is_multi = is_multi
|
||||
if is_multi then
|
||||
methods.__tostring = function(self)
|
||||
return tostring(self.name) .. "(" .. tostring(table.concat((function()
|
||||
local _accum_0 = { }
|
||||
local _len_0 = 1
|
||||
local _list_0 = self.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local v = _list_0[_index_0]
|
||||
_accum_0[_len_0] = v.map and v:map(fn) or v
|
||||
for _index_0 = 1, #self do
|
||||
local v = self[_index_0]
|
||||
_accum_0[_len_0] = repr(v)
|
||||
_len_0 = _len_0 + 1
|
||||
end
|
||||
return _accum_0
|
||||
end)())))
|
||||
end)(), ', ')) .. ")"
|
||||
end
|
||||
return self
|
||||
methods.map = function(self, fn)
|
||||
do
|
||||
local ret = fn(self)
|
||||
if ret then
|
||||
return ret
|
||||
end
|
||||
end
|
||||
local new_vals
|
||||
do
|
||||
local _accum_0 = { }
|
||||
local _len_0 = 1
|
||||
for _index_0 = 1, #self do
|
||||
local v = self[_index_0]
|
||||
_accum_0[_len_0] = v.map and v:map(fn) or v
|
||||
_len_0 = _len_0 + 1
|
||||
end
|
||||
new_vals = _accum_0
|
||||
end
|
||||
local ret = getmetatable(self)(unpack(new_vals))
|
||||
return ret
|
||||
end
|
||||
else
|
||||
methods.__tostring = function(self)
|
||||
return tostring(self.name) .. "(" .. tostring(repr(self.value)) .. ")"
|
||||
end
|
||||
methods.map = function(self, fn)
|
||||
return fn(self) or self
|
||||
end
|
||||
end
|
||||
end
|
||||
if is_multi then
|
||||
Types[name] = immutable(nil, methods)
|
||||
else
|
||||
Types[name] = immutable({
|
||||
"value",
|
||||
"source"
|
||||
"value"
|
||||
}, methods)
|
||||
end
|
||||
end
|
||||
Tree("Nomsu", {
|
||||
Tree("EscapedNomsu", 'single', {
|
||||
as_lua = function(self, nomsu)
|
||||
return Lua.Value(self.source, "nomsu:parse(Nomsu(", repr(self.value.source), ", ", repr(tostring(self.value:as_nomsu(true))), "))")
|
||||
local make_tree
|
||||
make_tree = function(t)
|
||||
if type(t) ~= 'userdata' then
|
||||
return repr(t)
|
||||
end
|
||||
if t.is_multi then
|
||||
local bits
|
||||
do
|
||||
local _accum_0 = { }
|
||||
local _len_0 = 1
|
||||
for _index_0 = 1, #t do
|
||||
local bit = t[_index_0]
|
||||
_accum_0[_len_0] = make_tree(bit)
|
||||
_len_0 = _len_0 + 1
|
||||
end
|
||||
bits = _accum_0
|
||||
end
|
||||
return t.type .. "(" .. table.concat(bits, ", ") .. ")"
|
||||
else
|
||||
return t.type .. "(" .. make_tree(t.value) .. ")"
|
||||
end
|
||||
end
|
||||
return Lua.Value(nil, make_tree(self.value))
|
||||
end,
|
||||
as_nomsu = function(self, inline)
|
||||
if inline == nil then
|
||||
@ -79,15 +112,18 @@ Tree("Nomsu", {
|
||||
local nomsu = self.value:as_nomsu(true)
|
||||
if nomsu == nil and not inline then
|
||||
nomsu = self.value:as_nomsu()
|
||||
return nomsu and Nomsu(self.source, "\\:\n ", nomsu)
|
||||
return nomsu and Nomsu(nil, "\\:\n ", nomsu)
|
||||
end
|
||||
return nomsu and Nomsu(self.source, "\\(", nomsu, ")")
|
||||
return nomsu and Nomsu(nil, "\\(", nomsu, ")")
|
||||
end,
|
||||
map = function(self, fn)
|
||||
return fn(self) or self:map(fn)
|
||||
end
|
||||
})
|
||||
Tree("Block", {
|
||||
Tree("Block", 'multi', {
|
||||
as_lua = function(self, nomsu)
|
||||
local lua = Lua(self.source)
|
||||
for i, line in ipairs(self.value) do
|
||||
local lua = Lua()
|
||||
for i, line in ipairs(self) do
|
||||
local line_lua = line:as_lua(nomsu)
|
||||
if i > 1 then
|
||||
lua:append("\n")
|
||||
@ -101,8 +137,8 @@ Tree("Block", {
|
||||
inline = false
|
||||
end
|
||||
if inline then
|
||||
local nomsu = Nomsu(self.source)
|
||||
for i, line in ipairs(self.value) do
|
||||
local nomsu = Nomsu()
|
||||
for i, line in ipairs(self) do
|
||||
if i > 1 then
|
||||
nomsu:append("; ")
|
||||
end
|
||||
@ -114,11 +150,11 @@ Tree("Block", {
|
||||
end
|
||||
return nomsu
|
||||
end
|
||||
local nomsu = Nomsu(self.source)
|
||||
for i, line in ipairs(self.value) do
|
||||
local nomsu = Nomsu()
|
||||
for i, line in ipairs(self) do
|
||||
line = assert(line:as_nomsu(nil, true), "Could not convert line to nomsu")
|
||||
nomsu:append(line)
|
||||
if i < #self.value then
|
||||
if i < #self then
|
||||
nomsu:append("\n")
|
||||
if tostring(line):match("\n") then
|
||||
nomsu:append("\n")
|
||||
@ -129,7 +165,7 @@ Tree("Block", {
|
||||
end
|
||||
})
|
||||
local math_expression = re.compile([[ ([+-] " ")* "%" (" " [*/^+-] (" " [+-])* " %")+ !. ]])
|
||||
Tree("Action", {
|
||||
Tree("Action", 'multi', {
|
||||
as_lua = function(self, nomsu)
|
||||
local stub = self:get_stub()
|
||||
local compile_action = nomsu.environment.COMPILE_ACTIONS[stub]
|
||||
@ -138,9 +174,8 @@ Tree("Action", {
|
||||
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]
|
||||
for _index_0 = 1, #self do
|
||||
local arg = self[_index_0]
|
||||
if arg.type ~= "Word" then
|
||||
_accum_0[_len_0] = arg
|
||||
_len_0 = _len_0 + 1
|
||||
@ -166,30 +201,29 @@ Tree("Action", {
|
||||
return ret
|
||||
end
|
||||
local action = rawget(nomsu.environment.ACTIONS, stub)
|
||||
local lua = Lua.Value(self.source)
|
||||
local lua = Lua.Value()
|
||||
if not action and math_expression:match(stub) then
|
||||
for i, tok in ipairs(self.value) do
|
||||
for i, tok in ipairs(self) 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)))
|
||||
error("non-expression value inside math expression: " .. tostring(colored.yellow(repr(tok))))
|
||||
end
|
||||
if tok.type == "Action" then
|
||||
tok_lua:parenthesize()
|
||||
end
|
||||
lua:append(tok_lua)
|
||||
end
|
||||
if i < #self.value then
|
||||
if i < #self then
|
||||
lua:append(" ")
|
||||
end
|
||||
end
|
||||
return lua
|
||||
end
|
||||
local args = { }
|
||||
for i, tok in ipairs(self.value) do
|
||||
for i, tok in ipairs(self) do
|
||||
local _continue_0 = false
|
||||
repeat
|
||||
if tok.type == "Word" then
|
||||
@ -198,8 +232,7 @@ Tree("Action", {
|
||||
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)
|
||||
error("Cannot use:\n" .. tostring(colored.yellow(repr(tok))) .. "\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
|
||||
@ -240,9 +273,8 @@ Tree("Action", {
|
||||
do
|
||||
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]
|
||||
for _index_0 = 1, #self do
|
||||
local t = self[_index_0]
|
||||
_accum_0[_len_0] = (t.type == "Word" and t.value or "%" .. tostring(t.value))
|
||||
_len_0 = _len_0 + 1
|
||||
end
|
||||
@ -252,9 +284,8 @@ Tree("Action", {
|
||||
do
|
||||
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]
|
||||
for _index_0 = 1, #self do
|
||||
local t = self[_index_0]
|
||||
_accum_0[_len_0] = (t.type == "Word" and t.value or "%")
|
||||
_len_0 = _len_0 + 1
|
||||
end
|
||||
@ -271,8 +302,8 @@ Tree("Action", {
|
||||
can_use_colon = false
|
||||
end
|
||||
if inline then
|
||||
local nomsu = Nomsu(self.source)
|
||||
for i, bit in ipairs(self.value) do
|
||||
local nomsu = Nomsu()
|
||||
for i, bit in ipairs(self) do
|
||||
if bit.type == "Word" then
|
||||
if i > 1 then
|
||||
nomsu:append(" ")
|
||||
@ -294,10 +325,10 @@ Tree("Action", {
|
||||
end
|
||||
return nomsu
|
||||
else
|
||||
local nomsu = Nomsu(self.source)
|
||||
local nomsu = Nomsu()
|
||||
local next_space = ""
|
||||
local last_colon = nil
|
||||
for i, bit in ipairs(self.value) do
|
||||
for i, bit in ipairs(self) do
|
||||
if bit.type == "Word" then
|
||||
nomsu:append(next_space, bit.value)
|
||||
next_space = " "
|
||||
@ -331,9 +362,9 @@ Tree("Action", {
|
||||
end
|
||||
if bit.type ~= "List" and bit.type ~= "Dict" and bit.type ~= "Text" then
|
||||
if i == 1 then
|
||||
arg_nomsu = Nomsu(bit.source, "(..)\n ", arg_nomsu)
|
||||
arg_nomsu = Nomsu(nil, "(..)\n ", arg_nomsu)
|
||||
else
|
||||
arg_nomsu = Nomsu(bit.source, "\n ", arg_nomsu)
|
||||
arg_nomsu = Nomsu(nil, "\n ", arg_nomsu)
|
||||
end
|
||||
end
|
||||
if last_colon == i - 1 and (bit.type == "Action" or bit.type == "Block") then
|
||||
@ -351,15 +382,14 @@ Tree("Action", {
|
||||
end
|
||||
end
|
||||
})
|
||||
Tree("Text", {
|
||||
Tree("Text", 'multi', {
|
||||
as_lua = function(self, nomsu)
|
||||
local lua = Lua.Value(self.source)
|
||||
local lua = Lua.Value()
|
||||
local string_buffer = ""
|
||||
local _list_0 = self.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
for _index_0 = 1, #self do
|
||||
local _continue_0 = false
|
||||
repeat
|
||||
local bit = _list_0[_index_0]
|
||||
local bit = self[_index_0]
|
||||
if type(bit) == "string" then
|
||||
string_buffer = string_buffer .. bit
|
||||
_continue_0 = true
|
||||
@ -374,14 +404,13 @@ Tree("Text", {
|
||||
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)
|
||||
error("Cannot use " .. tostring(colored.yellow(repr(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, ")")
|
||||
bit_lua = Lua.Value(nil, "stringify(", bit_lua, ")")
|
||||
end
|
||||
lua:append(bit_lua)
|
||||
_continue_0 = true
|
||||
@ -406,10 +435,9 @@ Tree("Text", {
|
||||
inline = false
|
||||
end
|
||||
if inline then
|
||||
local nomsu = Nomsu(self.source, '"')
|
||||
local _list_0 = self.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local bit = _list_0[_index_0]
|
||||
local nomsu = Nomsu(nil, '"')
|
||||
for _index_0 = 1, #self do
|
||||
local bit = self[_index_0]
|
||||
if type(bit) == 'string' then
|
||||
nomsu:append((bit:gsub("\\", "\\\\"):gsub("\n", "\\n")))
|
||||
else
|
||||
@ -431,8 +459,8 @@ Tree("Text", {
|
||||
if inline_version and #inline_version <= MAX_LINE then
|
||||
return inline_version
|
||||
end
|
||||
local nomsu = Nomsu(self.source, '".."\n ')
|
||||
for i, bit in ipairs(self.value) do
|
||||
local nomsu = Nomsu(nil, '".."\n ')
|
||||
for i, bit in ipairs(self) do
|
||||
if type(bit) == 'string' then
|
||||
nomsu:append((bit:gsub("\\", "\\\\"):gsub("\n", "\n ")))
|
||||
else
|
||||
@ -448,7 +476,7 @@ Tree("Text", {
|
||||
return nil
|
||||
end
|
||||
nomsu:append("\\\n ", interp_nomsu)
|
||||
if i < #self.value then
|
||||
if i < #self then
|
||||
nomsu:append("\n ..")
|
||||
end
|
||||
end
|
||||
@ -458,15 +486,14 @@ Tree("Text", {
|
||||
end
|
||||
end
|
||||
})
|
||||
Tree("List", {
|
||||
Tree("List", 'multi', {
|
||||
as_lua = function(self, nomsu)
|
||||
local lua = Lua.Value(self.source, "{")
|
||||
local lua = Lua.Value(nil, "{")
|
||||
local line_length = 0
|
||||
for i, item in ipairs(self.value) do
|
||||
for i, item in ipairs(self) 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)
|
||||
error("Cannot use " .. tostring(colored.yellow(repr(item))) .. " as a list item, since it's not an expression.", 0)
|
||||
end
|
||||
lua:append(item_lua)
|
||||
local item_string = tostring(item_lua)
|
||||
@ -476,7 +503,7 @@ Tree("List", {
|
||||
else
|
||||
line_length = line_length + #last_line
|
||||
end
|
||||
if i < #self.value then
|
||||
if i < #self then
|
||||
if line_length >= MAX_LINE then
|
||||
lua:append(",\n ")
|
||||
line_length = 0
|
||||
@ -494,8 +521,8 @@ Tree("List", {
|
||||
inline = false
|
||||
end
|
||||
if inline then
|
||||
local nomsu = Nomsu(self.source, "[")
|
||||
for i, item in ipairs(self.value) do
|
||||
local nomsu = Nomsu(nil, "[")
|
||||
for i, item in ipairs(self) do
|
||||
local item_nomsu = item:as_nomsu(true)
|
||||
if not (item_nomsu) then
|
||||
return nil
|
||||
@ -512,11 +539,10 @@ Tree("List", {
|
||||
if inline_version and #inline_version <= MAX_LINE then
|
||||
return inline_version
|
||||
end
|
||||
local nomsu = Nomsu(self.source, "[..]")
|
||||
local line = Nomsu(self.source, "\n ")
|
||||
local _list_0 = self.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local item = _list_0[_index_0]
|
||||
local nomsu = Nomsu(nil, "[..]")
|
||||
local line = Nomsu(nil, "\n ")
|
||||
for _index_0 = 1, #self do
|
||||
local item = self[_index_0]
|
||||
local item_nomsu = item:as_nomsu(true)
|
||||
if item_nomsu and #line + #", " + #item_nomsu <= MAX_LINE then
|
||||
if #line.bits > 1 then
|
||||
@ -532,7 +558,7 @@ Tree("List", {
|
||||
end
|
||||
if #line.bits > 1 then
|
||||
nomsu:append(line)
|
||||
line = Nomsu(item.source, "\n ")
|
||||
line = Nomsu(nil, "\n ")
|
||||
end
|
||||
line:append(item_nomsu)
|
||||
end
|
||||
@ -544,11 +570,11 @@ Tree("List", {
|
||||
end
|
||||
end
|
||||
})
|
||||
Tree("Dict", {
|
||||
Tree("Dict", 'multi', {
|
||||
as_lua = function(self, nomsu)
|
||||
local lua = Lua.Value(self.source, "{")
|
||||
local lua = Lua.Value(nil, "{")
|
||||
local line_length = 0
|
||||
for i, entry in ipairs(self.value) do
|
||||
for i, entry in ipairs(self) do
|
||||
local entry_lua = entry:as_lua(nomsu)
|
||||
lua:append(entry_lua)
|
||||
local entry_lua_str = tostring(entry_lua)
|
||||
@ -558,7 +584,7 @@ Tree("Dict", {
|
||||
else
|
||||
line_length = line_length + #entry_lua_str
|
||||
end
|
||||
if i < #self.value then
|
||||
if i < #self then
|
||||
if line_length >= MAX_LINE then
|
||||
lua:append(",\n ")
|
||||
line_length = 0
|
||||
@ -576,8 +602,8 @@ Tree("Dict", {
|
||||
inline = false
|
||||
end
|
||||
if inline then
|
||||
local nomsu = Nomsu(self.source, "{")
|
||||
for i, entry in ipairs(self.value) do
|
||||
local nomsu = Nomsu(nil, "{")
|
||||
for i, entry in ipairs(self) do
|
||||
local entry_nomsu = entry:as_nomsu(true)
|
||||
if not (entry_nomsu) then
|
||||
return nil
|
||||
@ -594,11 +620,10 @@ Tree("Dict", {
|
||||
if inline_version then
|
||||
return inline_version
|
||||
end
|
||||
local nomsu = Nomsu(self.source, "{..}")
|
||||
local line = Nomsu(self.source, "\n ")
|
||||
local _list_0 = self.value
|
||||
for _index_0 = 1, #_list_0 do
|
||||
local entry = _list_0[_index_0]
|
||||
local nomsu = Nomsu(nil, "{..}")
|
||||
local line = Nomsu(nil, "\n ")
|
||||
for _index_0 = 1, #self do
|
||||
local entry = self[_index_0]
|
||||
local entry_nomsu = entry:as_nomsu()
|
||||
if not (entry_nomsu) then
|
||||
return nil
|
||||
@ -611,7 +636,7 @@ Tree("Dict", {
|
||||
else
|
||||
if #line.bits > 1 then
|
||||
nomsu:append(line)
|
||||
line = Nomsu(bit.source, "\n ")
|
||||
line = Nomsu(nil, "\n ")
|
||||
end
|
||||
line:append(entry_nomsu)
|
||||
end
|
||||
@ -623,33 +648,31 @@ Tree("Dict", {
|
||||
end
|
||||
end
|
||||
})
|
||||
Tree("DictEntry", {
|
||||
Tree("DictEntry", 'multi', {
|
||||
as_lua = function(self, nomsu)
|
||||
local key, value = self.value[1], self.value[2]
|
||||
local key, value = self[1], self[2]
|
||||
local key_lua = 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)
|
||||
error("Cannot use " .. tostring(colored.yellow(repr(key))) .. " as a dict key, since it's not an expression.", 0)
|
||||
end
|
||||
local value_lua = value and value:as_lua(nomsu) or Lua.Value(key.source, "true")
|
||||
local value_lua = value and value:as_lua(nomsu) or Lua.Value(nil, "true")
|
||||
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)
|
||||
error("Cannot use " .. tostring(colored.yellow(repr(value))) .. " 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
|
||||
return Lua(key.source, key_str, "=", value_lua)
|
||||
return Lua(nil, key_str, "=", value_lua)
|
||||
elseif tostring(key_lua):sub(1, 1) == "[" then
|
||||
return Lua(key.source, "[ ", key_lua, "]=", value_lua)
|
||||
return Lua(nil, "[ ", key_lua, "]=", value_lua)
|
||||
else
|
||||
return Lua(key.source, "[", key_lua, "]=", value_lua)
|
||||
return Lua(nil, "[", key_lua, "]=", value_lua)
|
||||
end
|
||||
end,
|
||||
as_nomsu = function(self, inline)
|
||||
if inline == nil then
|
||||
inline = true
|
||||
end
|
||||
local key, value = self.value[1], self.value[2]
|
||||
local key, value = self[1], self[2]
|
||||
local key_nomsu = key:as_nomsu(true)
|
||||
if not (key_nomsu) then
|
||||
return nil
|
||||
@ -661,7 +684,7 @@ Tree("DictEntry", {
|
||||
if value then
|
||||
value_nomsu = value:as_nomsu(true)
|
||||
else
|
||||
value_nomsu = Nomsu(key.source, "")
|
||||
value_nomsu = Nomsu(nil, "")
|
||||
end
|
||||
if inline and not value_nomsu then
|
||||
return nil
|
||||
@ -675,43 +698,35 @@ Tree("DictEntry", {
|
||||
return nil
|
||||
end
|
||||
end
|
||||
return Nomsu(key.source, key_nomsu, ":", value_nomsu)
|
||||
return Nomsu(nil, key_nomsu, ":", value_nomsu)
|
||||
end
|
||||
})
|
||||
Tree("IndexChain", {
|
||||
Tree("IndexChain", 'multi', {
|
||||
as_lua = function(self, nomsu)
|
||||
local lua = self.value[1]:as_lua(nomsu)
|
||||
local lua = self[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)
|
||||
error("Cannot index " .. tostring(colored.yellow(repr(self[1]))) .. ", since it's not an expression.", 0)
|
||||
end
|
||||
local first_char = tostring(lua):sub(1, 1)
|
||||
if first_char == "{" or first_char == '"' or first_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
|
||||
for i = 2, #self do
|
||||
local key = self[i]
|
||||
local key_lua = 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 an index, since it's not an expression.", 0)
|
||||
error("Cannot use " .. tostring(colored.yellow(repr(key))) .. " as an index, since it's not an expression.", 0)
|
||||
end
|
||||
if tostring(key_lua):sub(1, 1) == '[' then
|
||||
lua:append("[ ", key_lua, "]")
|
||||
local key_lua_str = tostring(key_lua)
|
||||
do
|
||||
local lua_id = key_lua_str:match("^['\"]([a-zA-Z_][a-zA-Z0-9_]*)['\"]$")
|
||||
if lua_id then
|
||||
lua:append("." .. tostring(lua_id))
|
||||
elseif key_lua_str: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
|
||||
@ -720,8 +735,8 @@ Tree("IndexChain", {
|
||||
if inline == nil then
|
||||
inline = false
|
||||
end
|
||||
local nomsu = Nomsu(self.source)
|
||||
for i, bit in ipairs(self.value) do
|
||||
local nomsu = Nomsu()
|
||||
for i, bit in ipairs(self) do
|
||||
if i > 1 then
|
||||
nomsu:append(".")
|
||||
end
|
||||
@ -737,18 +752,18 @@ Tree("IndexChain", {
|
||||
return nomsu
|
||||
end
|
||||
})
|
||||
Tree("Number", {
|
||||
Tree("Number", 'single', {
|
||||
as_lua = function(self, nomsu)
|
||||
return Lua.Value(self.source, tostring(self.value))
|
||||
return Lua.Value(nil, tostring(self.value))
|
||||
end,
|
||||
as_nomsu = function(self, inline)
|
||||
if inline == nil then
|
||||
inline = false
|
||||
end
|
||||
return Nomsu(self.source, tostring(self.value))
|
||||
return Nomsu(nil, tostring(self.value))
|
||||
end
|
||||
})
|
||||
Tree("Var", {
|
||||
Tree("Var", 'single', {
|
||||
as_lua_id = function(v)
|
||||
return "_" .. (v:gsub("%W", function(c)
|
||||
if c == "_" then
|
||||
@ -759,16 +774,16 @@ Tree("Var", {
|
||||
end))
|
||||
end,
|
||||
as_lua = function(self, nomsu)
|
||||
return Lua.Value(self.source, self.as_lua_id(self.value))
|
||||
return Lua.Value(nil, self.as_lua_id(self.value))
|
||||
end,
|
||||
as_nomsu = function(self, inline)
|
||||
if inline == nil then
|
||||
inline = false
|
||||
end
|
||||
return Nomsu(self.source, "%", self.value)
|
||||
return Nomsu(nil, "%", self.value)
|
||||
end
|
||||
})
|
||||
Tree("Word", {
|
||||
Tree("Word", 'single', {
|
||||
as_lua = function(self, nomsu)
|
||||
return error("Attempt to convert Word to lua")
|
||||
end,
|
||||
@ -776,12 +791,12 @@ Tree("Word", {
|
||||
if inline == nil then
|
||||
inline = false
|
||||
end
|
||||
return Nomsu(self.source, self.value)
|
||||
return Nomsu(nil, self.value)
|
||||
end
|
||||
})
|
||||
Tree("Comment", {
|
||||
Tree("Comment", 'single', {
|
||||
as_lua = function(self, nomsu)
|
||||
return Lua(self.source, "--" .. self.value:gsub("\n", "\n--") .. "\n")
|
||||
return Lua(nil, "--" .. self.value:gsub("\n", "\n--") .. "\n")
|
||||
end,
|
||||
as_nomsu = function(self, inline)
|
||||
if inline == nil then
|
||||
@ -791,9 +806,9 @@ Tree("Comment", {
|
||||
return nil
|
||||
end
|
||||
if self.value:match("\n") then
|
||||
return Nomsu(self.source, "#..", self.value:gsub("\n", "\n "))
|
||||
return Nomsu(nil, "#..", self.value:gsub("\n", "\n "))
|
||||
else
|
||||
return Nomsu(self.source, "#", self.value)
|
||||
return Nomsu(nil, "#", self.value)
|
||||
end
|
||||
end
|
||||
})
|
||||
|
245
nomsu_tree.moon
245
nomsu_tree.moon
@ -13,46 +13,57 @@ Types.is_node = (n)->
|
||||
type(n) == 'userdata' and getmetatable(n) and Types[n.type] == getmetatable(n)
|
||||
|
||||
-- Helper method:
|
||||
Tree = (name, methods)->
|
||||
Tree = (name, kind, methods)->
|
||||
assert((kind == 'single') or (kind == 'multi'))
|
||||
is_multi = (kind == 'multi')
|
||||
with methods
|
||||
.__tostring = => "#{@name}(#{repr(@value)}, #{repr @source})"
|
||||
.with_value = (value)=> getmetatable(self)(value, @source)
|
||||
.with_value = (value)=> getmetatable(self)(value)
|
||||
.type = name
|
||||
.name = name
|
||||
.original_nomsu = =>
|
||||
leading_space = 0
|
||||
src_file = FILE_CACHE[@source.filename]
|
||||
while src_file\sub(@source.start-leading_space-1, @source.start-leading_space-1) == " "
|
||||
leading_space += 1
|
||||
if src_file\sub(@source.start-leading_space-1, @source.start-leading_space-1) != "\n"
|
||||
leading_space = 0
|
||||
ret = tostring(@source\get_text!)\gsub("\n"..((" ")\rep(leading_space)), "\n")
|
||||
return ret
|
||||
.is_multi = is_multi
|
||||
if is_multi
|
||||
.__tostring = => "#{@name}(#{table.concat [repr(v) for v in *@], ', '})"
|
||||
.map = (fn)=>
|
||||
if mapped = fn(self)
|
||||
return mapped
|
||||
if Tuple\is_instance(@value)
|
||||
return @with_value(Tuple(unpack([v.map and v\map(fn) or v for v in *@value])))
|
||||
return self
|
||||
if ret = fn(@)
|
||||
return ret
|
||||
new_vals = [v.map and v\map(fn) or v for v in *@]
|
||||
ret = getmetatable(self)(unpack(new_vals))
|
||||
return ret
|
||||
else
|
||||
.__tostring = => "#{@name}(#{repr(@value)})"
|
||||
.map = (fn)=> fn(@) or @
|
||||
|
||||
Types[name] = immutable {"value","source"}, methods
|
||||
if is_multi
|
||||
Types[name] = immutable nil, methods
|
||||
else
|
||||
Types[name] = immutable {"value"}, methods
|
||||
|
||||
|
||||
Tree "Nomsu",
|
||||
Tree "EscapedNomsu", 'single',
|
||||
as_lua: (nomsu)=>
|
||||
Lua.Value(@source, "nomsu:parse(Nomsu(",repr(@value.source),", ",repr(tostring(@value\as_nomsu(true))),"))")
|
||||
make_tree = (t)->
|
||||
if type(t) != 'userdata'
|
||||
return repr(t)
|
||||
if t.is_multi
|
||||
bits = [make_tree(bit) for bit in *t]
|
||||
return t.type.."("..table.concat(bits, ", ")..")"
|
||||
else
|
||||
return t.type.."("..make_tree(t.value)..")"
|
||||
Lua.Value nil, make_tree(@value)
|
||||
|
||||
as_nomsu: (inline=false)=>
|
||||
nomsu = @value\as_nomsu(true)
|
||||
if nomsu == nil and not inline
|
||||
nomsu = @value\as_nomsu!
|
||||
return nomsu and Nomsu(@source, "\\:\n ", nomsu)
|
||||
return nomsu and Nomsu(@source, "\\(", nomsu, ")")
|
||||
return nomsu and Nomsu nil, "\\:\n ", nomsu
|
||||
return nomsu and Nomsu nil, "\\(", nomsu, ")"
|
||||
|
||||
Tree "Block",
|
||||
map: (fn)=> fn(@) or @\map(fn)
|
||||
|
||||
Tree "Block", 'multi',
|
||||
as_lua: (nomsu)=>
|
||||
lua = Lua(@source)
|
||||
for i,line in ipairs @value
|
||||
lua = Lua!
|
||||
for i,line in ipairs @
|
||||
line_lua = line\as_lua(nomsu)
|
||||
if i > 1
|
||||
lua\append "\n"
|
||||
@ -61,31 +72,31 @@ Tree "Block",
|
||||
|
||||
as_nomsu: (inline=false)=>
|
||||
if inline
|
||||
nomsu = Nomsu(@source)
|
||||
for i,line in ipairs @value
|
||||
nomsu = Nomsu!
|
||||
for i,line in ipairs @
|
||||
if i > 1
|
||||
nomsu\append "; "
|
||||
line_nomsu = line\as_nomsu(true)
|
||||
return nil unless line_nomsu
|
||||
nomsu\append line_nomsu
|
||||
return nomsu
|
||||
nomsu = Nomsu(@source)
|
||||
for i, line in ipairs @value
|
||||
nomsu = Nomsu!
|
||||
for i, line in ipairs @
|
||||
line = assert(line\as_nomsu(nil, true), "Could not convert line to nomsu")
|
||||
nomsu\append line
|
||||
if i < #@value
|
||||
if i < #@
|
||||
nomsu\append "\n"
|
||||
if tostring(line)\match("\n")
|
||||
nomsu\append "\n"
|
||||
return nomsu
|
||||
|
||||
math_expression = re.compile [[ ([+-] " ")* "%" (" " [*/^+-] (" " [+-])* " %")+ !. ]]
|
||||
Tree "Action",
|
||||
Tree "Action", 'multi',
|
||||
as_lua: (nomsu)=>
|
||||
stub = @get_stub!
|
||||
compile_action = nomsu.environment.COMPILE_ACTIONS[stub]
|
||||
if compile_action
|
||||
args = [arg for arg in *@value when arg.type != "Word"]
|
||||
args = [arg for arg in *@ when arg.type != "Word"]
|
||||
-- Force all compile-time actions to take a tree location
|
||||
args = [args[p-1] for p in *nomsu.environment.ARG_ORDERS[compile_action][stub]]
|
||||
-- Force Lua to avoid tail call optimization for debugging purposes
|
||||
@ -93,33 +104,31 @@ Tree "Action",
|
||||
if not ret then error("Failed to produce any Lua")
|
||||
return ret
|
||||
action = rawget(nomsu.environment.ACTIONS, stub)
|
||||
lua = Lua.Value(@source)
|
||||
lua = Lua.Value!
|
||||
if not action and math_expression\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
|
||||
for i,tok in ipairs @
|
||||
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}")
|
||||
error("non-expression value inside math expression: #{colored.yellow repr(tok)}")
|
||||
if tok.type == "Action"
|
||||
tok_lua\parenthesize!
|
||||
lua\append tok_lua
|
||||
if i < #@value
|
||||
if i < #@
|
||||
lua\append " "
|
||||
return lua
|
||||
|
||||
args = {}
|
||||
for i, tok in ipairs @value
|
||||
for i, tok in ipairs @
|
||||
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
|
||||
error "Cannot use:\n#{colored.yellow repr(tok)}\nas an argument to #{stub}, since it's not an expression, it produces: #{repr arg_lua}", 0
|
||||
insert args, arg_lua
|
||||
|
||||
if action
|
||||
@ -136,14 +145,14 @@ Tree "Action",
|
||||
|
||||
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]
|
||||
[(t.type == "Word" and t.value or "%#{t.value}") for t in *@]
|
||||
else [(t.type == "Word" and t.value or "%") for t in *@]
|
||||
return concat(bits, " ")
|
||||
|
||||
as_nomsu: (inline=false, can_use_colon=false)=>
|
||||
if inline
|
||||
nomsu = Nomsu(@source)
|
||||
for i,bit in ipairs @value
|
||||
nomsu = Nomsu!
|
||||
for i,bit in ipairs @
|
||||
if bit.type == "Word"
|
||||
if i > 1
|
||||
nomsu\append " "
|
||||
@ -158,11 +167,11 @@ Tree "Action",
|
||||
nomsu\append arg_nomsu
|
||||
return nomsu
|
||||
else
|
||||
nomsu = Nomsu(@source)
|
||||
nomsu = Nomsu!
|
||||
next_space = ""
|
||||
-- TODO: track line length as we go and use 80-that instead of 80 for wrapping
|
||||
last_colon = nil
|
||||
for i,bit in ipairs @value
|
||||
for i,bit in ipairs @
|
||||
if bit.type == "Word"
|
||||
nomsu\append next_space, bit.value
|
||||
next_space = " "
|
||||
@ -189,9 +198,9 @@ Tree "Action",
|
||||
-- These types carry their own indentation
|
||||
if bit.type != "List" and bit.type != "Dict" and bit.type != "Text"
|
||||
if i == 1
|
||||
arg_nomsu = Nomsu(bit.source, "(..)\n ", arg_nomsu)
|
||||
arg_nomsu = Nomsu(nil, "(..)\n ", arg_nomsu)
|
||||
else
|
||||
arg_nomsu = Nomsu(bit.source, "\n ", arg_nomsu)
|
||||
arg_nomsu = Nomsu(nil, "\n ", arg_nomsu)
|
||||
|
||||
if last_colon == i-1 and (bit.type == "Action" or bit.type == "Block")
|
||||
next_space = ""
|
||||
@ -202,11 +211,11 @@ Tree "Action",
|
||||
next_space = "\n.."
|
||||
return nomsu
|
||||
|
||||
Tree "Text",
|
||||
Tree "Text", 'multi',
|
||||
as_lua: (nomsu)=>
|
||||
lua = Lua.Value(@source)
|
||||
lua = Lua.Value!
|
||||
string_buffer = ""
|
||||
for bit in *@value
|
||||
for bit in *@
|
||||
if type(bit) == "string"
|
||||
string_buffer ..= bit
|
||||
continue
|
||||
@ -216,11 +225,10 @@ Tree "Text",
|
||||
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
|
||||
error "Cannot use #{colored.yellow repr(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,")")
|
||||
bit_lua = Lua.Value(nil, "stringify(",bit_lua,")")
|
||||
lua\append bit_lua
|
||||
|
||||
if string_buffer ~= "" or #lua.bits == 0
|
||||
@ -233,8 +241,8 @@ Tree "Text",
|
||||
|
||||
as_nomsu: (inline=false)=>
|
||||
if inline
|
||||
nomsu = Nomsu(@source, '"')
|
||||
for bit in *@value
|
||||
nomsu = Nomsu(nil, '"')
|
||||
for bit in *@
|
||||
if type(bit) == 'string'
|
||||
-- TODO: unescape better?
|
||||
nomsu\append (bit\gsub("\\","\\\\")\gsub("\n","\\n"))
|
||||
@ -251,8 +259,8 @@ Tree "Text",
|
||||
inline_version = @as_nomsu(true)
|
||||
if inline_version and #inline_version <= MAX_LINE
|
||||
return inline_version
|
||||
nomsu = Nomsu(@source, '".."\n ')
|
||||
for i, bit in ipairs @value
|
||||
nomsu = Nomsu(nil, '".."\n ')
|
||||
for i, bit in ipairs @
|
||||
if type(bit) == 'string'
|
||||
nomsu\append (bit\gsub("\\","\\\\")\gsub("\n","\n "))
|
||||
else
|
||||
@ -265,19 +273,18 @@ Tree "Text",
|
||||
interp_nomsu = bit\as_nomsu!
|
||||
return nil unless interp_nomsu
|
||||
nomsu\append "\\\n ", interp_nomsu
|
||||
if i < #@value
|
||||
if i < #@
|
||||
nomsu\append "\n .."
|
||||
return nomsu
|
||||
|
||||
Tree "List",
|
||||
Tree "List", 'multi',
|
||||
as_lua: (nomsu)=>
|
||||
lua = Lua.Value @source, "{"
|
||||
lua = Lua.Value nil, "{"
|
||||
line_length = 0
|
||||
for i, item in ipairs @value
|
||||
for i, item in ipairs @
|
||||
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
|
||||
error "Cannot use #{colored.yellow repr(item)} as a list item, since it's not an expression.", 0
|
||||
lua\append item_lua
|
||||
item_string = tostring(item_lua)
|
||||
last_line = item_string\match("[^\n]*$")
|
||||
@ -285,7 +292,7 @@ Tree "List",
|
||||
line_length = #last_line
|
||||
else
|
||||
line_length += #last_line
|
||||
if i < #@value
|
||||
if i < #@
|
||||
if line_length >= MAX_LINE
|
||||
lua\append ",\n "
|
||||
line_length = 0
|
||||
@ -297,8 +304,8 @@ Tree "List",
|
||||
|
||||
as_nomsu: (inline=false)=>
|
||||
if inline
|
||||
nomsu = Nomsu(@source, "[")
|
||||
for i, item in ipairs @value
|
||||
nomsu = Nomsu(nil, "[")
|
||||
for i, item in ipairs @
|
||||
item_nomsu = item\as_nomsu(true)
|
||||
return nil unless item_nomsu
|
||||
if i > 1
|
||||
@ -310,9 +317,9 @@ Tree "List",
|
||||
inline_version = @as_nomsu(true)
|
||||
if inline_version and #inline_version <= MAX_LINE
|
||||
return inline_version
|
||||
nomsu = Nomsu(@source, "[..]")
|
||||
line = Nomsu(@source, "\n ")
|
||||
for item in *@value
|
||||
nomsu = Nomsu(nil, "[..]")
|
||||
line = Nomsu(nil, "\n ")
|
||||
for item in *@
|
||||
item_nomsu = item\as_nomsu(true)
|
||||
if item_nomsu and #line + #", " + #item_nomsu <= MAX_LINE
|
||||
if #line.bits > 1
|
||||
@ -324,17 +331,17 @@ Tree "List",
|
||||
return nil unless item_nomsu
|
||||
if #line.bits > 1
|
||||
nomsu\append line
|
||||
line = Nomsu(item.source, "\n ")
|
||||
line = Nomsu(nil, "\n ")
|
||||
line\append item_nomsu
|
||||
if #line.bits > 1
|
||||
nomsu\append line
|
||||
return nomsu
|
||||
|
||||
Tree "Dict",
|
||||
Tree "Dict", 'multi',
|
||||
as_lua: (nomsu)=>
|
||||
lua = Lua.Value @source, "{"
|
||||
lua = Lua.Value nil, "{"
|
||||
line_length = 0
|
||||
for i, entry in ipairs @value
|
||||
for i, entry in ipairs @
|
||||
entry_lua = entry\as_lua(nomsu)
|
||||
lua\append entry_lua
|
||||
entry_lua_str = tostring(entry_lua)
|
||||
@ -344,7 +351,7 @@ Tree "Dict",
|
||||
line_length = #last_line
|
||||
else
|
||||
line_length += #entry_lua_str
|
||||
if i < #@value
|
||||
if i < #@
|
||||
if line_length >= MAX_LINE
|
||||
lua\append ",\n "
|
||||
line_length = 0
|
||||
@ -356,8 +363,8 @@ Tree "Dict",
|
||||
|
||||
as_nomsu: (inline=false)=>
|
||||
if inline
|
||||
nomsu = Nomsu(@source, "{")
|
||||
for i, entry in ipairs @value
|
||||
nomsu = Nomsu(nil, "{")
|
||||
for i, entry in ipairs @
|
||||
entry_nomsu = entry\as_nomsu(true)
|
||||
return nil unless entry_nomsu
|
||||
if i > 1
|
||||
@ -368,9 +375,9 @@ Tree "Dict",
|
||||
else
|
||||
inline_version = @as_nomsu(true)
|
||||
if inline_version then return inline_version
|
||||
nomsu = Nomsu(@source, "{..}")
|
||||
line = Nomsu(@source, "\n ")
|
||||
for entry in *@value
|
||||
nomsu = Nomsu(nil, "{..}")
|
||||
line = Nomsu(nil, "\n ")
|
||||
for entry in *@
|
||||
entry_nomsu = entry\as_nomsu!
|
||||
return nil unless entry_nomsu
|
||||
if #line + #tostring(entry_nomsu) <= MAX_LINE
|
||||
@ -380,82 +387,78 @@ Tree "Dict",
|
||||
else
|
||||
if #line.bits > 1
|
||||
nomsu\append line
|
||||
line = Nomsu(bit.source, "\n ")
|
||||
line = Nomsu(nil, "\n ")
|
||||
line\append entry_nomsu
|
||||
if #line.bits > 1
|
||||
nomsu\append line
|
||||
return nomsu
|
||||
|
||||
Tree "DictEntry",
|
||||
Tree "DictEntry", 'multi',
|
||||
as_lua: (nomsu)=>
|
||||
key, value = @value[1], @value[2]
|
||||
key, value = @[1], @[2]
|
||||
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 a dict key, since it's not an expression.", 0
|
||||
value_lua = value and value\as_lua(nomsu) or Lua.Value(key.source, "true")
|
||||
error "Cannot use #{colored.yellow repr(key)} as a dict key, since it's not an expression.", 0
|
||||
value_lua = value and value\as_lua(nomsu) or Lua.Value(nil, "true")
|
||||
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
|
||||
error "Cannot use #{colored.yellow repr(value)} as a dict value, since it's not an expression.", 0
|
||||
key_str = tostring(key_lua)\match([=[["']([a-zA-Z_][a-zA-Z0-9_]*)['"]]=])
|
||||
return if key_str
|
||||
Lua key.source, key_str,"=",value_lua
|
||||
Lua nil, 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 key.source, "[ ",key_lua,"]=",value_lua
|
||||
Lua nil, "[ ",key_lua,"]=",value_lua
|
||||
else
|
||||
Lua key.source, "[",key_lua,"]=",value_lua
|
||||
Lua nil, "[",key_lua,"]=",value_lua
|
||||
|
||||
as_nomsu: (inline=true)=>
|
||||
key, value = @value[1], @value[2]
|
||||
key, value = @[1], @[2]
|
||||
key_nomsu = key\as_nomsu(true)
|
||||
return nil unless key_nomsu
|
||||
if key.type == "Action" or key.type == "Block"
|
||||
key_nomsu\parenthesize!
|
||||
value_nomsu = if value
|
||||
value\as_nomsu(true)
|
||||
else Nomsu(key.source, "")
|
||||
else Nomsu(nil, "")
|
||||
if inline and not value_nomsu then return nil
|
||||
if not value_nomsu
|
||||
return nil if inline
|
||||
value_nomsu = value\as_nomsu!
|
||||
return nil unless value_nomsu
|
||||
return Nomsu key.source, key_nomsu, ":", value_nomsu
|
||||
return Nomsu nil, key_nomsu, ":", value_nomsu
|
||||
|
||||
|
||||
Tree "IndexChain",
|
||||
Tree "IndexChain", 'multi',
|
||||
as_lua: (nomsu)=>
|
||||
lua = @value[1]\as_lua(nomsu)
|
||||
lua = @[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
|
||||
error "Cannot index #{colored.yellow repr(@[1])}, since it's not an expression.", 0
|
||||
first_char = tostring(lua)\sub(1,1)
|
||||
if first_char == "{" or first_char == '"' or first_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
|
||||
for i=2,#@
|
||||
key = @[i]
|
||||
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
|
||||
error "Cannot use #{colored.yellow repr(key)} as an index, since it's not an expression.", 0
|
||||
key_lua_str = tostring(key_lua)
|
||||
if lua_id = key_lua_str\match("^['\"]([a-zA-Z_][a-zA-Z0-9_]*)['\"]$")
|
||||
lua\append ".#{lua_id}"
|
||||
elseif key_lua_str\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"]
|
||||
if tostring(key_lua)\sub(1,1) == '['
|
||||
lua\append "[ ",key_lua,"]"
|
||||
lua\append "[ ",key_lua," ]"
|
||||
else
|
||||
lua\append "[",key_lua,"]"
|
||||
return lua
|
||||
|
||||
as_nomsu: (inline=false)=>
|
||||
nomsu = Nomsu(@source)
|
||||
for i, bit in ipairs @value
|
||||
nomsu = Nomsu!
|
||||
for i, bit in ipairs @
|
||||
if i > 1
|
||||
nomsu\append "."
|
||||
bit_nomsu = bit\as_nomsu(true)
|
||||
@ -465,39 +468,39 @@ Tree "IndexChain",
|
||||
nomsu\append bit_nomsu
|
||||
return nomsu
|
||||
|
||||
Tree "Number",
|
||||
Tree "Number", 'single',
|
||||
as_lua: (nomsu)=>
|
||||
Lua.Value(@source, tostring(@value))
|
||||
Lua.Value(nil, tostring(@value))
|
||||
|
||||
as_nomsu: (inline=false)=>
|
||||
return Nomsu(@source, tostring(@value))
|
||||
return Nomsu(nil, tostring(@value))
|
||||
|
||||
Tree "Var",
|
||||
Tree "Var", 'single',
|
||||
as_lua_id: (v)->
|
||||
"_"..(v\gsub("%W", (c)-> if c == "_" then "__" else ("_%x")\format(c\byte!)))
|
||||
|
||||
as_lua: (nomsu)=>
|
||||
Lua.Value(@source, self.as_lua_id(@value))
|
||||
Lua.Value(nil, self.as_lua_id(@value))
|
||||
|
||||
as_nomsu: (inline=false)=>
|
||||
return Nomsu(@source, "%", @value)
|
||||
return Nomsu(nil, "%", @value)
|
||||
|
||||
Tree "Word",
|
||||
Tree "Word", 'single',
|
||||
as_lua: (nomsu)=>
|
||||
error("Attempt to convert Word to lua")
|
||||
|
||||
as_nomsu: (inline=false)=>
|
||||
return Nomsu(@source, @value)
|
||||
return Nomsu(nil, @value)
|
||||
|
||||
Tree "Comment",
|
||||
Tree "Comment", 'single',
|
||||
as_lua: (nomsu)=>
|
||||
Lua(@source, "--"..@value\gsub("\n","\n--").."\n")
|
||||
Lua(nil, "--"..@value\gsub("\n","\n--").."\n")
|
||||
|
||||
as_nomsu: (inline=false)=>
|
||||
return nil if inline
|
||||
if @value\match("\n")
|
||||
return Nomsu(@source, "#..", @value\gsub("\n", "\n "))
|
||||
return Nomsu(nil, "#..", @value\gsub("\n", "\n "))
|
||||
else
|
||||
return Nomsu(@source, "#", @value)
|
||||
return Nomsu(nil, "#", @value)
|
||||
|
||||
return Types
|
||||
|
@ -42,7 +42,7 @@ try: foo 99
|
||||
|
||||
assume ((\(5 + 5) as value) = 10) or barf "%tree as value failed."
|
||||
|
||||
assume (((\(foo %x)'s source code) as text) = "foo %x") or barf "source code failed."
|
||||
assume (((\(foo %x) as nomsu) as text) = "foo %x") or barf "source code failed."
|
||||
|
||||
assume ((repr [1,2]) = "{1, 2}") or barf "repr failed."
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user