Initial working version.

This commit is contained in:
Bruce Hill 2018-05-16 18:12:56 -07:00
parent af9dc07025
commit 6f6c4377b2
11 changed files with 443 additions and 476 deletions

View File

@ -144,7 +144,7 @@ immediately
for %var in %start to %stop via %step %body for %var in %start to %stop via %step %body
..to ..to
# This uses Lua's approach of only allowing loop-scoped variables in a loop # 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 <-
Lua ".." Lua ".."
for \(%var as lua expr)=\(%start as lua expr),\(%stop as lua expr),\(%step as lua expr) do 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 %body has subtree % where
(%.type = "Action") and (%.type = "Action") and
((%'s stub) is "do next %") 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 "\n ::continue_\(%var as lua identifier)::;"
to %lua write "\nend --numeric for-loop" to %lua write "\nend --numeric for-loop"
@ -161,7 +161,7 @@ immediately
%body has subtree % where %body has subtree % where
(%.type = "Action") and (%.type = "Action") and
((%'s stub) is "stop %") and ((%'s stub) is "stop %") and
%.value.2.value is %var.value %.2 = %var
.. ..
%lua <- %lua <-
Lua ".." Lua ".."
@ -179,7 +179,7 @@ immediately
immediately immediately
compile [for %var in %iterable %body] to compile [for %var in %iterable %body] to
# This uses Lua's approach of only allowing loop-scoped variables in a loop # 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 <-
Lua ".." Lua ".."
for i,\(%var as lua identifier) in ipairs(\(%iterable as lua expr)) do for i,\(%var as lua identifier) in ipairs(\(%iterable as lua expr)) do
@ -188,14 +188,14 @@ immediately
%body has subtree % where %body has subtree % where
(%.type = "Action") and (%.type = "Action") and
((%'s stub) is "do next %") 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 (Lua "\n ::continue_\(%var as lua identifier)::;")
to %lua write "\nend --foreach-loop" to %lua write "\nend --foreach-loop"
if if
%body has subtree % where %body has subtree % where
(%.type = "Action") and (%.type = "Action") and
((%'s stub) is "stop %") and ((%'s stub) is "stop %") and
%.value.2.value is %var.value %.2 = %var
.. ..
%lua <- %lua <-
Lua ".." Lua ".."
@ -209,8 +209,8 @@ immediately
immediately immediately
compile [for %key = %value in %iterable %body] to compile [for %key = %value in %iterable %body] to
# This uses Lua's approach of only allowing loop-scoped variables in a loop # 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 (%key.type is "Var") or barf "Loop expected variable, not: \%key"
assume (%value.type is "Var") or barf "Loop expected variable, not: \(%value's source code)" assume (%value.type is "Var") or barf "Loop expected variable, not: \%value"
%lua <- %lua <-
Lua ".." Lua ".."
for \(%key as lua identifier),\(%value as lua identifier) in pairs(\(%iterable as lua expr)) do 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 %body has subtree % where
(%.type = "Action") and (%.type = "Action") and
((%'s stub) is "do next %") 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)::;") ..: to %lua write (Lua "\n ::continue_\(%key as lua identifier)::;")
if if
%body has subtree % where %body has subtree % where
(%.type = "Action") and (%.type = "Action") and
((%'s stub) is "do next %") 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 (Lua "\n ::continue_\(%value as lua identifier)::;")
to %lua write "\nend --foreach-loop" to %lua write "\nend --foreach-loop"
@ -235,14 +235,14 @@ immediately
%body has subtree % where %body has subtree % where
(%.type = "Action") and (%.type = "Action") and
((%'s stub) is "stop %") and ((%'s stub) is "stop %") and
%.value.2.value is %key.value %.2 = %key
..: to %stop_labels write "\n::stop_\(%key as lua identifier)::;" ..: to %stop_labels write "\n::stop_\(%key as lua identifier)::;"
if if
%body has subtree % where %body has subtree % where
(%.type = "Action") and (%.type = "Action") and
((%'s stub) is "stop %") and ((%'s stub) is "stop %") and
%.value.2.value is %value.value %.2 = %value
..: to %stop_labels write "\n::stop_\(%value as lua identifier)::;" ..: to %stop_labels write "\n::stop_\(%value as lua identifier)::;"
if: (length of %stop_labels) > 0 if: (length of %stop_labels) > 0
@ -262,15 +262,14 @@ immediately
%is_first <- (yes) %is_first <- (yes)
%seen_else <- (no) %seen_else <- (no)
%branches <- %branches <-
%body.value if (%body.type = "Block") else [%body] %body if (%body.type = "Block") else [%body]
for %func_call in %branches for %func_call in %branches
assume (%func_call.type is "Action") or barf ".." assume (%func_call.type is "Action") or barf ".."
Invalid format for 'when' statement. Only '*' blocks are allowed. Invalid format for 'when' statement. Only '*' blocks are allowed.
%tokens <- %func_call.value
with {..} with {..}
%star: %tokens.1 %star: %func_call.1
%condition: %tokens.2 %condition: %func_call.2
%action: %tokens.3 %action: %func_call.3
.. ..
assume ((%star and (%star.type is "Word")) and (%star.value is "*")) or barf ".." assume ((%star and (%star.type is "Word")) and (%star.value is "*")) or barf ".."
Invalid format for 'when' statement. Lines must begin with '*' Invalid format for 'when' statement. Lines must begin with '*'
@ -311,12 +310,11 @@ immediately
%is_first <- (yes) %is_first <- (yes)
%seen_else <- (no) %seen_else <- (no)
%branches <- %branches <-
%body.value if (%body.type = "Block") else [%body] %body if (%body.type = "Block") else [%body]
for %func_call in %branches for %func_call in %branches
assume (%func_call.type is "Action") or barf ".." assume (%func_call.type is "Action") or barf ".."
Invalid format for 'when' statement. Only '*' blocks are allowed. Invalid format for 'when' statement. Only '*' blocks are allowed.
%tokens <- %func_call.value with {%star:%func_call.1, %condition:%func_call.2, %action:%func_call.3}
with {%star:%tokens.1, %condition:%tokens.2, %action:%tokens.3}
assume ((%star and (%star.type is "Word")) and (%star.value is "*")) or barf ".." assume ((%star and (%star.type is "Word")) and (%star.value is "*")) or barf ".."
Invalid format for 'when' statement. Lines must begin with '*' Invalid format for 'when' statement. Lines must begin with '*'
assume %condition or barf ".." assume %condition or barf ".."

View File

@ -42,7 +42,7 @@ compile [all of %items, all %items] to
unless: (%items' "type") is "List" unless: (%items' "type") is "List"
return: Lua value "utils.all(\(%items as lua expr))" return: Lua value "utils.all(\(%items as lua expr))"
%clauses <- [] %clauses <- []
for % in (%items' "value") for % in %items
lua> "table.insert(\%clauses, \(% as lua expr));" lua> "table.insert(\%clauses, \(% as lua expr));"
return: Lua value "(\(%clauses joined with " and "))" return: Lua value "(\(%clauses joined with " and "))"
parse [not all of %items, not all %items] as: not (all of %items) 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" unless: (%items' "type") is "List"
return: Lua value "utils.any(\(%items as lua expr))" return: Lua value "utils.any(\(%items as lua expr))"
%clauses <- [] %clauses <- []
for % in (%items' "value") for % in %items
lua> "table.insert(\%clauses, \(% as lua expr));" lua> "table.insert(\%clauses, \(% as lua expr));"
return: Lua value "(\(%clauses joined with " or "))" return: Lua value "(\(%clauses joined with " or "))"
parse [none of %items, none %items] as: not (any of %items) 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" unless: (%items' "type") is "List"
return: Lua value "utils.sum(\(%items as lua expr))" return: Lua value "utils.sum(\(%items as lua expr))"
%clauses <- [] %clauses <- []
for % in (%items' "value") for % in %items
lua> "table.insert(\%clauses, \(% as lua expr));" lua> "table.insert(\%clauses, \(% as lua expr));"
return: Lua value "(\(%clauses joined with " + "))" return: Lua value "(\(%clauses joined with " + "))"
compile [product of %items, product %items] to compile [product of %items, product %items] to
unless: (%items' "type") is "List" unless: (%items' "type") is "List"
return: Lua value "utils.product(\(%items as lua expr))" return: Lua value "utils.product(\(%items as lua expr))"
%clauses <- [] %clauses <- []
for % in (%items' "value") for % in %items
lua> "table.insert(\%clauses, \(% as lua expr));" lua> "table.insert(\%clauses, \(% as lua expr));"
return: Lua value "(\(%clauses joined with " * "))" return: Lua value "(\(%clauses joined with " * "))"
action [avg of %items, average of %items] action [avg of %items, average of %items]

View File

@ -6,9 +6,9 @@
immediately immediately
lua> ".." lua> ".."
nomsu:define_compile_action("compile %actions to %lua", function(tree, \%actions, \%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 = {} local stubs = {}
for i, action in ipairs(\%actions.value) do for i, action in ipairs(\%actions) do
stubs[i] = action:get_stub(true) stubs[i] = action:get_stub(true)
end end
stubs = repr(stubs) stubs = repr(stubs)
@ -19,7 +19,7 @@ immediately
end end
lua:append("function(tree") lua:append("function(tree")
local args = {} 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 if tok.type == "Var" then args[#args+1] = tok:as_lua(nomsu) end
end end
for i, arg in ipairs(args) do for i, arg in ipairs(args) do
@ -36,9 +36,9 @@ immediately
immediately immediately
compile [action %actions %body] to compile [action %actions %body] to
lua> ".." lua> ".."
local lua = Lua(tree.source, "nomsu:define_action(") local lua = Lua(nil, "nomsu:define_action(")
local stubs = {} local stubs = {}
for i, action in ipairs(\%actions.value) do for i, action in ipairs(\%actions) do
stubs[i] = action:get_stub(true) stubs[i] = action:get_stub(true)
end end
stubs = repr(stubs) stubs = repr(stubs)
@ -49,7 +49,7 @@ immediately
end end
lua:append("function(") lua:append("function(")
local args = {} 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 if tok.type == "Var" then args[#args+1] = tok:as_lua(nomsu) end
end end
for i, arg in ipairs(args) do for i, arg in ipairs(args) do
@ -66,9 +66,9 @@ immediately
immediately immediately
compile [parse %shorthand as %longhand] to compile [parse %shorthand as %longhand] to
lua> ".." lua> ".."
local lua = Lua(tree.source, "nomsu:define_compile_action(") local lua = Lua(nil, "nomsu:define_compile_action(")
local stubs = {} local stubs = {}
for i, action in ipairs(\%shorthand.value) do for i, action in ipairs(\%shorthand) do
stubs[i] = action:get_stub(true) stubs[i] = action:get_stub(true)
end end
stubs = repr(stubs) stubs = repr(stubs)
@ -79,24 +79,24 @@ immediately
end end
lua:append("function(tree") lua:append("function(tree")
local replacements = {} 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 if tok.type == "Var" then
local lua_var = tostring(tok:as_lua(nomsu)) local lua_var = tostring(tok:as_lua(nomsu))
replacements[tok.value] = lua_var replacements[tok] = lua_var
lua:append(", ", lua_var) lua:append(", ", lua_var)
end end
end end
local function make_tree(t) 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 = {} local bits = {}
for i, entry in ipairs(t) do for i, entry in ipairs(t) do
bits[i] = make_tree(entry) bits[i] = make_tree(entry)
end end
return "Tuple("..table.concat(bits, ", ")..")" return t.type.."("..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]
else else
return t.type.."("..make_tree(t.value)..")" return t.type.."("..make_tree(t.value)..")"
end end
@ -118,6 +118,12 @@ action [remove action %stub]
ARG_ORDERS[fn] = nil ARG_ORDERS[fn] = nil
immediately immediately
action [%tree as nomsu]
=lua "\%tree:as_nomsu()"
action [%tree as inline nomsu]
=lua "\%tree:as_nomsu(true)"
action [%tree as lua] action [%tree as lua]
=lua "\%tree:as_lua(nomsu)" =lua "\%tree:as_lua(nomsu)"
@ -125,7 +131,7 @@ immediately
lua> ".." lua> ".."
local lua = \%tree:as_lua(nomsu) local lua = \%tree:as_lua(nomsu)
if not lua.is_value then 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 end
return lua return lua
@ -156,9 +162,6 @@ immediately
parse [to %var write %code] as: lua> "\%var:append(\%code);" parse [to %var write %code] as: lua> "\%var:append(\%code);"
immediately 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 [repr %obj] to: Lua value "repr(\(%obj as lua expr))"
compile [%obj as text] to: Lua value "tostring(\(%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))" compile [type of %obj] to: Lua value "type(\(%obj as lua expr))"
@ -170,31 +173,23 @@ immediately
# Compiler tools # Compiler tools
immediately immediately
compile [run %code] to 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 immediately
compile [show lua %block] to compile [show lua %block] to
lua> ".." lua> ".."
local \%lua = \%block:as_lua(nomsu); local \%lua = \%block:as_lua(nomsu);
return Lua(\%block.source, "print(", repr(tostring(\%lua)), ");"); return Lua(nil, "print(", repr(tostring(\%lua)), ");");
immediately immediately
compile [say %message] to compile [say %message] to
lua> ".." lua> ".."
if \%message.type == "Text" then if \%message.type == "Text" then
return Lua(tree.source, "print(", \(%message as lua expr), ");"); return Lua(nil, "print(", \(%message as lua expr), ");");
else else
return Lua(tree.source, "print(stringify(", \(%message as lua expr), "));"); return Lua(nil, "print(stringify(", \(%message as lua expr), "));");
end 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 # Return
immediately immediately
# Return statement is wrapped in a do..end block because Lua is unhappy if you # 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] to: Lua "error(nil, 0);"
compile [barf %msg] to: Lua "error(\(%msg as lua expr), 0);" compile [barf %msg] to: Lua "error(\(%msg as lua expr), 0);"
compile [assume %condition] to compile [assume %condition] to
lua> "local \%assumption = 'Assumption failed: '..tostring(\%condition.source:get_text());" lua> "local \%assumption = 'Assumption failed: '..tostring(\%condition:as_nomsu());"
return return
Lua ".." Lua ".."
if not \(%condition as lua expr) then if not \(%condition as lua expr) then

View File

@ -28,18 +28,18 @@ immediately
local safe = {Text=true, Number=true}; local safe = {Text=true, Number=true};
local a_lua, b_lua = \%a:as_lua(nomsu), \%b:as_lua(nomsu); local a_lua, b_lua = \%a:as_lua(nomsu), \%b:as_lua(nomsu);
if safe[\%a.type] or safe[\%b.type] then 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 else
return Lua.Value(tree.source, "utils.equivalent(", a_lua, ", ", b_lua, ")"); return Lua.Value(nil, "utils.equivalent(", a_lua, ", ", b_lua, ")");
end end
compile [%a isn't %b, %a is not %b, %a not= %b, %a != %b] to compile [%a isn't %b, %a is not %b, %a not= %b, %a != %b] to
lua> ".." lua> ".."
local safe = {Text=true, Number=true}; local safe = {Text=true, Number=true};
local a_lua, b_lua = \%a:as_lua(nomsu), \%b:as_lua(nomsu); local a_lua, b_lua = \%a:as_lua(nomsu), \%b:as_lua(nomsu);
if safe[\%a.type] or safe[\%b.type] then 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 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 end
# For strict identity checking, use (%x's id) is (%y's id) # 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)]" compile [%'s id, id of %] to: Lua value "nomsu.ids[\(% as lua expr)]"
@ -48,11 +48,11 @@ immediately
immediately immediately
compile [%var <- %value] to compile [%var <- %value] to
lua> "local \%var_lua = \%var:as_lua(nomsu);" 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);" 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> ".." lua> ".."
local lua = Lua(tree.source, \%var_lua, ' = ', \%value_lua, ';'); local lua = Lua(nil, \%var_lua, ' = ', \%value_lua, ';');
if \%var.type == 'Var' then if \%var.type == 'Var' then
lua:add_free_vars({\%var}); lua:add_free_vars({\%var});
end end
@ -62,11 +62,11 @@ immediately
# Simultaneous mutli-assignments like: x,y,z = 1,x,3; # Simultaneous mutli-assignments like: x,y,z = 1,x,3;
compile [<- %assignments] to compile [<- %assignments] to
assume ((%assignments' "type") is "Dict") or barf ".." 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> ".." lua> ".."
local lhs, rhs = Lua(tree.source), Lua(\%assignments.source); local lhs, rhs = Lua(), Lua();
for i, item in ipairs(\%assignments.value) do for i, item in ipairs(\%assignments) do
local target, value = item.value[1], item.value[2]; local target, value = item[1], item[2];
local target_lua = target:as_lua(nomsu); local target_lua = target:as_lua(nomsu);
if not target_lua.is_value then error("Invalid target for assignment: "..target:get_src()); end if not target_lua.is_value then error("Invalid target for assignment: "..target:get_src()); end
local value_lua = value:as_lua(nomsu); local value_lua = value:as_lua(nomsu);
@ -81,35 +81,35 @@ immediately
lhs:append(target_lua); lhs:append(target_lua);
rhs:append(value_lua); rhs:append(value_lua);
end end
return Lua(tree.source, lhs, " = ", rhs, ";"); return Lua(nil, lhs, " = ", rhs, ";");
immediately immediately
compile [external %var <- %value] to compile [external %var <- %value] to
%var_lua <- (%var as lua) %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) %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);" return: Lua "\(%var_lua) = \(%value_lua);"
compile [with external %externs %body] to compile [with external %externs %body] to
%body_lua <- (%body as lua statements) %body_lua <- (%body as lua statements)
lua> "\%body_lua:remove_free_vars(\(%externs.value));" lua> "\%body_lua:remove_free_vars(\(%externs));"
return %body_lua return %body_lua
compile [with %assignments %body] to compile [with %assignments %body] to
%lua <- (%body as lua statements) %lua <- (%body as lua statements)
lua> ".." lua> ".."
local lhs, rhs = Lua(tree.source), Lua(\%assignments.source); local lhs, rhs = Lua(), Lua();
local vars = {}; local vars = {};
for i, item in ipairs(\%assignments.value) do for i, item in ipairs(\%assignments) do
local target, value = item.value[1], item.value[2]; local target, value = item[1], item[2];
if not target.type == "Var" then 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 end
local target_lua = target:as_lua(nomsu); local target_lua = target:as_lua(nomsu);
local value_lua = value:as_lua(nomsu); local value_lua = value:as_lua(nomsu);
if not value_lua.is_value then 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 end
if target.type == "Var" then if target.type == "Var" then
lhs:add_free_vars({target}); lhs:add_free_vars({target});

View File

@ -27,7 +27,7 @@ lua> ".."
}; };
for name, e in pairs(escapes) do for name, e in pairs(escapes) do
local lua = "'"..e.."'"; 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 end
local colors = { local colors = {
["reset color"]="\\\\27[0m", bright="\\\\27[1m", dim="\\\\27[2m", underscore="\\\\27[4m", ["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 for name, c in pairs(colors) do
local color = "'"..c.."'"; local color = "'"..c.."'";
local reset = "'"..colors["reset color"].."'"; 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(\%) 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 end
end end

View File

@ -146,9 +146,6 @@ local Types = require("nomsu_tree")
local NOMSU_DEFS local NOMSU_DEFS
do do
local _with_0 = { } local _with_0 = { }
_with_0.Tuple = function(values)
return Tuple(unpack(values))
end
_with_0.nl = P("\r") ^ -1 * P("\n") _with_0.nl = P("\r") ^ -1 * P("\n")
_with_0.ws = S(" \t") _with_0.ws = S(" \t")
_with_0.tonumber = tonumber _with_0.tonumber = tonumber
@ -228,16 +225,8 @@ end
setmetatable(NOMSU_DEFS, { setmetatable(NOMSU_DEFS, {
__index = function(self, key) __index = function(self, key)
local make_node local make_node
make_node = function(start, value, stop) make_node = function(src, ...)
if type(value) == 'table' then return Types[key](...)
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
end end
self[key] = make_node self[key] = make_node
return make_node return make_node
@ -250,7 +239,7 @@ do
anon_def <- ({ident} (" "*) ":" anon_def <- ({ident} (" "*) ":"
{((%nl " "+ [^%nl]*)+) / ([^%nl]*)}) -> "%1 <- %2" {((%nl " "+ [^%nl]*)+) / ([^%nl]*)}) -> "%1 <- %2"
captured_def <- ({ident} (" "*) "(" {ident} ")" (" "*) ":" captured_def <- ({ident} (" "*) "(" {ident} ")" (" "*) ":"
{((%nl " "+ [^%nl]*)+) / ([^%nl]*)}) -> "%1 <- ({} %3 {}) -> %2" {((%nl " "+ [^%nl]*)+) / ([^%nl]*)}) -> "%1 <- (({} %3) -> %2)"
ident <- [a-zA-Z_][a-zA-Z0-9_]* ident <- [a-zA-Z_][a-zA-Z0-9_]*
comment <- "--" [^%nl]* comment <- "--" [^%nl]*
]]) ]])
@ -430,12 +419,6 @@ do
run_lua = function(self, lua) run_lua = function(self, lua)
assert(type(lua) ~= 'string', "Attempt to run lua string instead of Lua (object)") assert(type(lua) ~= 'string', "Attempt to run lua string instead of Lua (object)")
local lua_string = tostring(lua) 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) local run_lua_fn, err = load(lua_string, filename, "t", self.environment)
if not run_lua_fn then if not run_lua_fn then
local n = 1 local n = 1
@ -450,10 +433,10 @@ do
return run_lua_fn() return run_lua_fn()
end, end,
tree_to_value = function(self, tree) tree_to_value = function(self, tree)
if tree.type == 'Text' and #tree.value == 1 and type(tree.value[1]) == 'string' then if tree.type == 'Text' and #tree == 1 and type(tree[1]) == 'string' then
return tree.value[1] return tree[1]
end 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) return self:run_lua(lua)
end, end,
walk_tree = function(self, tree, depth) walk_tree = function(self, tree, depth)
@ -464,10 +447,9 @@ do
if not (Types.is_node(tree)) then if not (Types.is_node(tree)) then
return return
end end
if Tuple:is_instance(tree.value) then if tree.is_multi then
local _list_0 = tree.value for _index_0 = 1, #tree do
for _index_0 = 1, #_list_0 do local v = tree[_index_0]
local v = _list_0[_index_0]
self:walk_tree(v, depth + 1) self:walk_tree(v, depth + 1)
end end
else else
@ -502,7 +484,7 @@ do
local lua = _block:as_lua(nomsu):as_statements() local lua = _block:as_lua(nomsu):as_statements()
lua:declare_locals() lua:declare_locals()
nomsu:run_lua(lua) nomsu:run_lua(lua)
return Lua(self.source, "if IMMEDIATE then\n ", lua, "\nend") return Lua(nil, "if IMMEDIATE then\n ", lua, "\nend")
end) end)
local add_lua_string_bits local add_lua_string_bits
add_lua_string_bits = function(lua, code) add_lua_string_bits = function(lua, code)
@ -510,46 +492,42 @@ do
lua:append(", ", code:as_lua(nomsu)) lua:append(", ", code:as_lua(nomsu))
return return
end end
local _list_0 = code.value for _index_0 = 1, #code do
for _index_0 = 1, #_list_0 do local bit = code[_index_0]
local bit = _list_0[_index_0]
lua:append(", ") lua:append(", ")
if type(bit) == "string" then if type(bit) == "string" then
lua:append(repr(bit)) lua:append(repr(bit))
else else
local bit_lua = bit:as_lua(nomsu) local bit_lua = bit:as_lua(nomsu)
if not (bit_lua.is_value) then if not (bit_lua.is_value) then
local line, src = bit.source:get_line(), bit.source:get_text() error("Cannot use " .. tostring(colored.yellow(repr(bit))) .. " as a string interpolation value, since it's not an expression.")
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(src)) .. " as a string interpolation value, since it's not an expression.")
end end
lua:append(bit_lua) lua:append(bit_lua)
end end
end end
end end
self:define_compile_action("Lua %code", function(self, _code) 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) add_lua_string_bits(lua, _code)
lua:append(")") lua:append(")")
return lua return lua
end) end)
self:define_compile_action("Lua value %code", function(self, _code) 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) add_lua_string_bits(lua, _code)
lua:append(")") lua:append(")")
return lua return lua
end) end)
local add_lua_bits local add_lua_bits
add_lua_bits = function(lua, code) add_lua_bits = function(lua, code)
local _list_0 = code.value for _index_0 = 1, #code do
for _index_0 = 1, #_list_0 do local bit = code[_index_0]
local bit = _list_0[_index_0]
if type(bit) == "string" then if type(bit) == "string" then
lua:append(bit) lua:append(bit)
else else
local bit_lua = bit:as_lua(nomsu) local bit_lua = bit:as_lua(nomsu)
if not (bit_lua.is_value) then if not (bit_lua.is_value) then
local line, src = bit.source:get_line(), bit.source:get_text() error("Cannot use " .. tostring(colored.yellow(repr(bit))) .. " as a string interpolation value, since it's not an expression.", 0)
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(src)) .. " as a string interpolation value, since it's not an expression.", 0)
end end
lua:append(bit_lua) lua:append(bit_lua)
end end
@ -558,20 +536,20 @@ do
end end
self:define_compile_action("lua> %code", function(self, _code) self:define_compile_action("lua> %code", function(self, _code)
if _code.type ~= "Text" then 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 end
return add_lua_bits(Lua(_code.source), _code) return add_lua_bits(Lua(), _code)
end) end)
self:define_compile_action("=lua %code", function(self, _code) self:define_compile_action("=lua %code", function(self, _code)
if _code.type ~= "Text" then 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 end
return add_lua_bits(Lua.Value(_code.source), _code) return add_lua_bits(Lua.Value(), _code)
end) end)
return self:define_compile_action("use %path", function(self, _path) return self:define_compile_action("use %path", function(self, _path)
local path = nomsu:tree_to_value(_path) local path = nomsu:tree_to_value(_path)
nomsu:run_file(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)
end end
} }

View File

@ -129,8 +129,6 @@ Types = require "nomsu_tree"
NOMSU_DEFS = with {} NOMSU_DEFS = with {}
-- Newline supports either windows-style CR+LF or unix-style LF -- Newline supports either windows-style CR+LF or unix-style LF
.Tuple = (values)->
return Tuple(unpack(values))
.nl = P("\r")^-1 * P("\n") .nl = P("\r")^-1 * P("\n")
.ws = S(" \t") .ws = S(" \t")
.tonumber = tonumber .tonumber = tonumber
@ -196,15 +194,8 @@ NOMSU_DEFS = with {}
return true return true
setmetatable(NOMSU_DEFS, {__index:(key)=> setmetatable(NOMSU_DEFS, {__index:(key)=>
make_node = (start, value, stop)-> make_node = (src, ...)->
if type(value) == 'table' then error("Not a tuple: #{repr value}")-- = Tuple(value) Types[key](...)
--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
self[key] = make_node self[key] = make_node
return make_node return make_node
}) })
@ -218,7 +209,7 @@ NOMSU_PATTERN = do
anon_def <- ({ident} (" "*) ":" anon_def <- ({ident} (" "*) ":"
{((%nl " "+ [^%nl]*)+) / ([^%nl]*)}) -> "%1 <- %2" {((%nl " "+ [^%nl]*)+) / ([^%nl]*)}) -> "%1 <- %2"
captured_def <- ({ident} (" "*) "(" {ident} ")" (" "*) ":" captured_def <- ({ident} (" "*) "(" {ident} ")" (" "*) ":"
{((%nl " "+ [^%nl]*)+) / ([^%nl]*)}) -> "%1 <- ({} %3 {}) -> %2" {((%nl " "+ [^%nl]*)+) / ([^%nl]*)}) -> "%1 <- (({} %3) -> %2)"
ident <- [a-zA-Z_][a-zA-Z0-9_]* ident <- [a-zA-Z_][a-zA-Z0-9_]*
comment <- "--" [^%nl]* comment <- "--" [^%nl]*
]] ]]
@ -401,13 +392,6 @@ class NomsuCompiler
run_lua: (lua)=> run_lua: (lua)=>
assert(type(lua) != 'string', "Attempt to run lua string instead of Lua (object)") assert(type(lua) != 'string', "Attempt to run lua string instead of Lua (object)")
lua_string = tostring(lua) 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) run_lua_fn, err = load(lua_string, filename, "t", @environment)
if not run_lua_fn if not run_lua_fn
n = 1 n = 1
@ -420,16 +404,16 @@ class NomsuCompiler
tree_to_value: (tree)=> tree_to_value: (tree)=>
-- Special case for text literals -- Special case for text literals
if tree.type == 'Text' and #tree.value == 1 and type(tree.value[1]) == 'string' if tree.type == 'Text' and #tree == 1 and type(tree[1]) == 'string'
return tree.value[1] return tree[1]
lua = Lua(tree.source, "return ",tree\as_lua(@),";") lua = Lua(nil, "return ",tree\as_lua(@),";")
return @run_lua(lua) return @run_lua(lua)
walk_tree: (tree, depth=0)=> walk_tree: (tree, depth=0)=>
coroutine.yield(tree, depth) coroutine.yield(tree, depth)
return unless Types.is_node(tree) return unless Types.is_node(tree)
if Tuple\is_instance(tree.value) if tree.is_multi
for v in *tree.value for v in *tree
@walk_tree(v, depth+1) @walk_tree(v, depth+1)
else else
@walk_tree(v, depth+1) @walk_tree(v, depth+1)
@ -451,61 +435,59 @@ class NomsuCompiler
lua = _block\as_lua(nomsu)\as_statements! lua = _block\as_lua(nomsu)\as_statements!
lua\declare_locals! lua\declare_locals!
nomsu\run_lua(lua) 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)-> add_lua_string_bits = (lua, code)->
if code.type != "Text" if code.type != "Text"
lua\append ", ", code\as_lua(nomsu) lua\append ", ", code\as_lua(nomsu)
return return
for bit in *code.value for bit in *code
lua\append ", " lua\append ", "
if type(bit) == "string" if type(bit) == "string"
lua\append repr(bit) lua\append repr(bit)
else else
bit_lua = bit\as_lua(nomsu) bit_lua = bit\as_lua(nomsu)
unless bit_lua.is_value unless bit_lua.is_value
line, src = bit.source\get_line!, bit.source\get_text! error "Cannot use #{colored.yellow repr(bit)} as a string interpolation value, since it's not an expression."
error "#{line}: Cannot use #{colored.yellow src} as a string interpolation value, since it's not an expression."
lua\append bit_lua lua\append bit_lua
@define_compile_action "Lua %code", (_code)=> @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) add_lua_string_bits(lua, _code)
lua\append ")" lua\append ")"
return lua return lua
@define_compile_action "Lua value %code", (_code)=> @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) add_lua_string_bits(lua, _code)
lua\append ")" lua\append ")"
return lua return lua
add_lua_bits = (lua, code)-> add_lua_bits = (lua, code)->
for bit in *code.value for bit in *code
if type(bit) == "string" if type(bit) == "string"
lua\append bit lua\append bit
else else
bit_lua = bit\as_lua(nomsu) bit_lua = bit\as_lua(nomsu)
unless bit_lua.is_value unless bit_lua.is_value
line, src = bit.source\get_line!, bit.source\get_text! error "Cannot use #{colored.yellow repr(bit)} as a string interpolation value, since it's not an expression.", 0
error "#{line}: Cannot use #{colored.yellow src} as a string interpolation value, since it's not an expression.", 0
lua\append bit_lua lua\append bit_lua
return lua return lua
@define_compile_action "lua> %code", (_code)=> @define_compile_action "lua> %code", (_code)=>
if _code.type != "Text" if _code.type != "Text"
return Lua @source, "nomsu:run_lua(", _code\as_lua(nomsu), ");" return Lua nil, "nomsu:run_lua(", _code\as_lua(nomsu), ");"
return add_lua_bits(Lua(_code.source), _code) return add_lua_bits(Lua!, _code)
@define_compile_action "=lua %code", (_code)=> @define_compile_action "=lua %code", (_code)=>
if _code.type != "Text" if _code.type != "Text"
return Lua.Value @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 '))"
return add_lua_bits(Lua.Value(_code.source), _code) return add_lua_bits(Lua.Value!, _code)
@define_compile_action "use %path", (_path)=> @define_compile_action "use %path", (_path)=>
path = nomsu\tree_to_value(_path) path = nomsu\tree_to_value(_path)
nomsu\run_file(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: -- 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 if arg and debug_getinfo(2).func != require

View File

@ -11,18 +11,16 @@ statement: (action / expression) (eol / (({} ([^%nl]* -> "Error while parsing li
inline_statement: inline_action / inline_expression inline_statement: inline_action / inline_expression
inline_block (Block): inline_block (Block):
{| inline_statement (%ws* ";" %ws* inline_statement)+ |} -> Tuple inline_statement (%ws* ";" %ws* inline_statement)+
block (Block): 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 inline_nomsu (EscapedNomsu): "\" noindex_inline_expression
indented_nomsu (Nomsu): indented_nomsu (EscapedNomsu):
"\" (noindex_inline_expression / (":" %ws* (inline_block / inline_action / inline_expression) eol) / indented_expression) "\" (noindex_inline_expression / (":" %ws* (inline_block / inline_action / inline_expression) eol) / indented_expression)
index_chain (IndexChain): index_chain (IndexChain):
{|
noindex_inline_expression ("." (text_word / noindex_inline_expression))+ noindex_inline_expression ("." (text_word / noindex_inline_expression))+
|} -> Tuple
noindex_inline_expression: noindex_inline_expression:
number / variable / inline_text / inline_list / inline_dict / inline_nomsu number / variable / inline_text / inline_list / inline_dict / inline_nomsu
@ -47,30 +45,30 @@ expression:
-- Function calls need at least one word in them -- Function calls need at least one word in them
inline_action (Action): inline_action (Action):
{| (inline_expression %ws*)* word (%ws* (inline_expression / word))* (inline_expression %ws*)* word (%ws* (inline_expression / word))*
(%ws* ":" %ws* (inline_block / inline_action / inline_expression))?|} -> Tuple (%ws* ":" %ws* (inline_block / inline_action / inline_expression))?
action (Action): 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) } word (Word): { %operator / (!number plain_word) }
text_word (Text): {| {%operator / (!number plain_word)} |} -> Tuple text_word (Text): {%operator / (!number plain_word)}
inline_text (Text): inline_text (Text):
!('".."' eol) !('".."' eol)
'"' ({| '"' (
({~ (('\"' -> '"') / ('\\' -> '\') / %escaped_char / [^%nl\"])+ ~} ({~ (('\"' -> '"') / ('\\' -> '\') / %escaped_char / [^%nl\"])+ ~}
/ inline_text_interpolation)* / 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 "#" -- Have to use "%indent" instead of "indent" etc. to avoid messing up text lines that start with "#"
indented_text (Text): indented_text (Text):
'".."' eol %nl ({| '".."' eol %nl (
{~ (%nl*) (%indent -> "") ~} {~ (%nl*) (%indent -> "") ~}
({~ ({~
(("\\" -> "\") / (("\" nodent "..") -> "")/ (%nl+ {~ %nodent -> "" ~}) / [^%nl\] / (!text_interpolation "\"))+ (("\\" -> "\") / (("\" nodent "..") -> "")/ (%nl+ {~ %nodent -> "" ~}) / [^%nl\] / (!text_interpolation "\"))+
~} / 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: inline_text_interpolation:
"\" ( "\" (
variable / inline_list / inline_dict / inline_text variable / inline_list / inline_dict / inline_text
@ -93,12 +91,11 @@ variable (Var): "%" { plain_word? }
inline_list (List): 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)) ("]" / (({} ([^%nl]*->"Failed to find a closing ] on the same line")) => error))
indented_list (List): indented_list (List):
"[..]" indent ({| "[..]" indent
list_line (nodent list_line)* list_line (nodent list_line)*
|} -> Tuple)
(dedent / (({} (non_dedent_error -> "Error while parsing list")) => error)) (dedent / (({} (non_dedent_error -> "Error while parsing list")) => error))
list_line: list_line:
((action / expression) !comma) ((action / expression) !comma)
@ -107,21 +104,20 @@ inline_list_item: inline_block / inline_action / inline_expression
inline_dict (Dict): 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) / (({} (%ws* comma? (!. / &%nl)->"Failed to find a closing } on the same line")) => error)
/ (({} ([^%nl]*->"Error while parsing dictionary")) => error)) / (({} ([^%nl]*->"Error while parsing dictionary")) => error))
indented_dict (Dict): indented_dict (Dict):
"{..}" indent ({| "{..}" indent
dict_line (nodent dict_line)* dict_line (nodent dict_line)*
|} -> Tuple)
(dedent / (({} (non_dedent_error -> "Error while parsing dict")) => error)) (dedent / (({} (non_dedent_error -> "Error while parsing dict")) => error))
dict_line: dict_line:
(dict_entry !comma) / (inline_dict_entry (comma dict_line?)?) (dict_entry !comma) / (inline_dict_entry (comma dict_line?)?)
dict_entry(DictEntry): dict_entry(DictEntry):
{| dict_key %ws* ":" %ws* (action / expression) |} -> Tuple dict_key %ws* ":" %ws* (action / expression)
inline_dict_entry(DictEntry): 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: dict_key:
text_word / inline_expression text_word / inline_expression

View File

@ -18,59 +18,92 @@ Types.is_node = function(n)
return type(n) == 'userdata' and getmetatable(n) and Types[n.type] == getmetatable(n) return type(n) == 'userdata' and getmetatable(n) and Types[n.type] == getmetatable(n)
end end
local Tree local Tree
Tree = function(name, methods) Tree = function(name, kind, methods)
assert((kind == 'single') or (kind == 'multi'))
local is_multi = (kind == 'multi')
do do
methods.__tostring = function(self)
return tostring(self.name) .. "(" .. tostring(repr(self.value)) .. ", " .. tostring(repr(self.source)) .. ")"
end
methods.with_value = function(self, value) methods.with_value = function(self, value)
return getmetatable(self)(value, self.source) return getmetatable(self)(value)
end end
methods.type = name methods.type = name
methods.name = name methods.name = name
methods.original_nomsu = function(self) methods.is_multi = is_multi
local leading_space = 0 if is_multi then
local src_file = FILE_CACHE[self.source.filename] methods.__tostring = function(self)
while src_file:sub(self.source.start - leading_space - 1, self.source.start - leading_space - 1) == " " do return tostring(self.name) .. "(" .. tostring(table.concat((function()
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()
local _accum_0 = { } local _accum_0 = { }
local _len_0 = 1 local _len_0 = 1
local _list_0 = self.value for _index_0 = 1, #self do
for _index_0 = 1, #_list_0 do local v = self[_index_0]
local v = _list_0[_index_0] _accum_0[_len_0] = repr(v)
_accum_0[_len_0] = v.map and v:map(fn) or v
_len_0 = _len_0 + 1 _len_0 = _len_0 + 1
end end
return _accum_0 return _accum_0
end)()))) end)(), ', ')) .. ")"
end end
return self methods.map = function(self, fn)
do
local ret = fn(self)
if ret then
return ret
end end
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({ Types[name] = immutable({
"value", "value"
"source"
}, methods) }, methods)
end end
Tree("Nomsu", { end
Tree("EscapedNomsu", 'single', {
as_lua = function(self, nomsu) 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, end,
as_nomsu = function(self, inline) as_nomsu = function(self, inline)
if inline == nil then if inline == nil then
@ -79,15 +112,18 @@ Tree("Nomsu", {
local nomsu = self.value:as_nomsu(true) local nomsu = self.value:as_nomsu(true)
if nomsu == nil and not inline then if nomsu == nil and not inline then
nomsu = self.value:as_nomsu() nomsu = self.value:as_nomsu()
return nomsu and Nomsu(self.source, "\\:\n ", nomsu) return nomsu and Nomsu(nil, "\\:\n ", nomsu)
end 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 end
}) })
Tree("Block", { Tree("Block", 'multi', {
as_lua = function(self, nomsu) as_lua = function(self, nomsu)
local lua = Lua(self.source) local lua = Lua()
for i, line in ipairs(self.value) do for i, line in ipairs(self) do
local line_lua = line:as_lua(nomsu) local line_lua = line:as_lua(nomsu)
if i > 1 then if i > 1 then
lua:append("\n") lua:append("\n")
@ -101,8 +137,8 @@ Tree("Block", {
inline = false inline = false
end end
if inline then if inline then
local nomsu = Nomsu(self.source) local nomsu = Nomsu()
for i, line in ipairs(self.value) do for i, line in ipairs(self) do
if i > 1 then if i > 1 then
nomsu:append("; ") nomsu:append("; ")
end end
@ -114,11 +150,11 @@ Tree("Block", {
end end
return nomsu return nomsu
end end
local nomsu = Nomsu(self.source) local nomsu = Nomsu()
for i, line in ipairs(self.value) do for i, line in ipairs(self) do
line = assert(line:as_nomsu(nil, true), "Could not convert line to nomsu") line = assert(line:as_nomsu(nil, true), "Could not convert line to nomsu")
nomsu:append(line) nomsu:append(line)
if i < #self.value then if i < #self then
nomsu:append("\n") nomsu:append("\n")
if tostring(line):match("\n") then if tostring(line):match("\n") then
nomsu:append("\n") nomsu:append("\n")
@ -129,7 +165,7 @@ Tree("Block", {
end end
}) })
local math_expression = re.compile([[ ([+-] " ")* "%" (" " [*/^+-] (" " [+-])* " %")+ !. ]]) local math_expression = re.compile([[ ([+-] " ")* "%" (" " [*/^+-] (" " [+-])* " %")+ !. ]])
Tree("Action", { Tree("Action", 'multi', {
as_lua = function(self, nomsu) as_lua = function(self, nomsu)
local stub = self:get_stub() local stub = self:get_stub()
local compile_action = nomsu.environment.COMPILE_ACTIONS[stub] local compile_action = nomsu.environment.COMPILE_ACTIONS[stub]
@ -138,9 +174,8 @@ Tree("Action", {
do do
local _accum_0 = { } local _accum_0 = { }
local _len_0 = 1 local _len_0 = 1
local _list_0 = self.value for _index_0 = 1, #self do
for _index_0 = 1, #_list_0 do local arg = self[_index_0]
local arg = _list_0[_index_0]
if arg.type ~= "Word" then if arg.type ~= "Word" then
_accum_0[_len_0] = arg _accum_0[_len_0] = arg
_len_0 = _len_0 + 1 _len_0 = _len_0 + 1
@ -166,30 +201,29 @@ Tree("Action", {
return ret return ret
end end
local action = rawget(nomsu.environment.ACTIONS, stub) 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 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 if tok.type == "Word" then
lua:append(tok.value) lua:append(tok.value)
else else
local tok_lua = tok:as_lua(nomsu) local tok_lua = tok:as_lua(nomsu)
if not (tok_lua.is_value) then if not (tok_lua.is_value) then
local src = tok.source:get_text() error("non-expression value inside math expression: " .. tostring(colored.yellow(repr(tok))))
error("non-expression value inside math expression: " .. tostring(colored.yellow(src)))
end end
if tok.type == "Action" then if tok.type == "Action" then
tok_lua:parenthesize() tok_lua:parenthesize()
end end
lua:append(tok_lua) lua:append(tok_lua)
end end
if i < #self.value then if i < #self then
lua:append(" ") lua:append(" ")
end end
end end
return lua return lua
end end
local args = { } local args = { }
for i, tok in ipairs(self.value) do for i, tok in ipairs(self) do
local _continue_0 = false local _continue_0 = false
repeat repeat
if tok.type == "Word" then if tok.type == "Word" then
@ -198,8 +232,7 @@ Tree("Action", {
end end
local arg_lua = tok:as_lua(nomsu) local arg_lua = tok:as_lua(nomsu)
if not (arg_lua.is_value) then if not (arg_lua.is_value) then
local line, src = tok.source:get_line(), tok.source:get_text() 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)
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 end
insert(args, arg_lua) insert(args, arg_lua)
_continue_0 = true _continue_0 = true
@ -240,9 +273,8 @@ Tree("Action", {
do do
local _accum_0 = { } local _accum_0 = { }
local _len_0 = 1 local _len_0 = 1
local _list_0 = self.value for _index_0 = 1, #self do
for _index_0 = 1, #_list_0 do local t = self[_index_0]
local t = _list_0[_index_0]
_accum_0[_len_0] = (t.type == "Word" and t.value or "%" .. tostring(t.value)) _accum_0[_len_0] = (t.type == "Word" and t.value or "%" .. tostring(t.value))
_len_0 = _len_0 + 1 _len_0 = _len_0 + 1
end end
@ -252,9 +284,8 @@ Tree("Action", {
do do
local _accum_0 = { } local _accum_0 = { }
local _len_0 = 1 local _len_0 = 1
local _list_0 = self.value for _index_0 = 1, #self do
for _index_0 = 1, #_list_0 do local t = self[_index_0]
local t = _list_0[_index_0]
_accum_0[_len_0] = (t.type == "Word" and t.value or "%") _accum_0[_len_0] = (t.type == "Word" and t.value or "%")
_len_0 = _len_0 + 1 _len_0 = _len_0 + 1
end end
@ -271,8 +302,8 @@ Tree("Action", {
can_use_colon = false can_use_colon = false
end end
if inline then if inline then
local nomsu = Nomsu(self.source) local nomsu = Nomsu()
for i, bit in ipairs(self.value) do for i, bit in ipairs(self) do
if bit.type == "Word" then if bit.type == "Word" then
if i > 1 then if i > 1 then
nomsu:append(" ") nomsu:append(" ")
@ -294,10 +325,10 @@ Tree("Action", {
end end
return nomsu return nomsu
else else
local nomsu = Nomsu(self.source) local nomsu = Nomsu()
local next_space = "" local next_space = ""
local last_colon = nil local last_colon = nil
for i, bit in ipairs(self.value) do for i, bit in ipairs(self) do
if bit.type == "Word" then if bit.type == "Word" then
nomsu:append(next_space, bit.value) nomsu:append(next_space, bit.value)
next_space = " " next_space = " "
@ -331,9 +362,9 @@ Tree("Action", {
end end
if bit.type ~= "List" and bit.type ~= "Dict" and bit.type ~= "Text" then if bit.type ~= "List" and bit.type ~= "Dict" and bit.type ~= "Text" then
if i == 1 then if i == 1 then
arg_nomsu = Nomsu(bit.source, "(..)\n ", arg_nomsu) arg_nomsu = Nomsu(nil, "(..)\n ", arg_nomsu)
else else
arg_nomsu = Nomsu(bit.source, "\n ", arg_nomsu) arg_nomsu = Nomsu(nil, "\n ", arg_nomsu)
end end
end end
if last_colon == i - 1 and (bit.type == "Action" or bit.type == "Block") then if last_colon == i - 1 and (bit.type == "Action" or bit.type == "Block") then
@ -351,15 +382,14 @@ Tree("Action", {
end end
end end
}) })
Tree("Text", { Tree("Text", 'multi', {
as_lua = function(self, nomsu) as_lua = function(self, nomsu)
local lua = Lua.Value(self.source) local lua = Lua.Value()
local string_buffer = "" local string_buffer = ""
local _list_0 = self.value for _index_0 = 1, #self do
for _index_0 = 1, #_list_0 do
local _continue_0 = false local _continue_0 = false
repeat repeat
local bit = _list_0[_index_0] local bit = self[_index_0]
if type(bit) == "string" then if type(bit) == "string" then
string_buffer = string_buffer .. bit string_buffer = string_buffer .. bit
_continue_0 = true _continue_0 = true
@ -374,14 +404,13 @@ Tree("Text", {
end end
local bit_lua = bit:as_lua(nomsu) local bit_lua = bit:as_lua(nomsu)
if not (bit_lua.is_value) then if not (bit_lua.is_value) then
local line, src = bit.source:get_line(), bit.source:get_text() error("Cannot use " .. tostring(colored.yellow(repr(bit))) .. " as a string interpolation value, since it's not an expression.", 0)
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(bit)) .. " as a string interpolation value, since it's not an expression.", 0)
end end
if #lua.bits > 0 then if #lua.bits > 0 then
lua:append("..") lua:append("..")
end end
if bit.type ~= "Text" then if bit.type ~= "Text" then
bit_lua = Lua.Value(bit.source, "stringify(", bit_lua, ")") bit_lua = Lua.Value(nil, "stringify(", bit_lua, ")")
end end
lua:append(bit_lua) lua:append(bit_lua)
_continue_0 = true _continue_0 = true
@ -406,10 +435,9 @@ Tree("Text", {
inline = false inline = false
end end
if inline then if inline then
local nomsu = Nomsu(self.source, '"') local nomsu = Nomsu(nil, '"')
local _list_0 = self.value for _index_0 = 1, #self do
for _index_0 = 1, #_list_0 do local bit = self[_index_0]
local bit = _list_0[_index_0]
if type(bit) == 'string' then if type(bit) == 'string' then
nomsu:append((bit:gsub("\\", "\\\\"):gsub("\n", "\\n"))) nomsu:append((bit:gsub("\\", "\\\\"):gsub("\n", "\\n")))
else else
@ -431,8 +459,8 @@ Tree("Text", {
if inline_version and #inline_version <= MAX_LINE then if inline_version and #inline_version <= MAX_LINE then
return inline_version return inline_version
end end
local nomsu = Nomsu(self.source, '".."\n ') local nomsu = Nomsu(nil, '".."\n ')
for i, bit in ipairs(self.value) do for i, bit in ipairs(self) do
if type(bit) == 'string' then if type(bit) == 'string' then
nomsu:append((bit:gsub("\\", "\\\\"):gsub("\n", "\n "))) nomsu:append((bit:gsub("\\", "\\\\"):gsub("\n", "\n ")))
else else
@ -448,7 +476,7 @@ Tree("Text", {
return nil return nil
end end
nomsu:append("\\\n ", interp_nomsu) nomsu:append("\\\n ", interp_nomsu)
if i < #self.value then if i < #self then
nomsu:append("\n ..") nomsu:append("\n ..")
end end
end end
@ -458,15 +486,14 @@ Tree("Text", {
end end
end end
}) })
Tree("List", { Tree("List", 'multi', {
as_lua = function(self, nomsu) as_lua = function(self, nomsu)
local lua = Lua.Value(self.source, "{") local lua = Lua.Value(nil, "{")
local line_length = 0 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) local item_lua = item:as_lua(nomsu)
if not (item_lua.is_value) then if not (item_lua.is_value) then
local line, src = item.source:get_line(), item.source:get_text() error("Cannot use " .. tostring(colored.yellow(repr(item))) .. " as a list item, since it's not an expression.", 0)
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(src)) .. " as a list item, since it's not an expression.", 0)
end end
lua:append(item_lua) lua:append(item_lua)
local item_string = tostring(item_lua) local item_string = tostring(item_lua)
@ -476,7 +503,7 @@ Tree("List", {
else else
line_length = line_length + #last_line line_length = line_length + #last_line
end end
if i < #self.value then if i < #self then
if line_length >= MAX_LINE then if line_length >= MAX_LINE then
lua:append(",\n ") lua:append(",\n ")
line_length = 0 line_length = 0
@ -494,8 +521,8 @@ Tree("List", {
inline = false inline = false
end end
if inline then if inline then
local nomsu = Nomsu(self.source, "[") local nomsu = Nomsu(nil, "[")
for i, item in ipairs(self.value) do for i, item in ipairs(self) do
local item_nomsu = item:as_nomsu(true) local item_nomsu = item:as_nomsu(true)
if not (item_nomsu) then if not (item_nomsu) then
return nil return nil
@ -512,11 +539,10 @@ Tree("List", {
if inline_version and #inline_version <= MAX_LINE then if inline_version and #inline_version <= MAX_LINE then
return inline_version return inline_version
end end
local nomsu = Nomsu(self.source, "[..]") local nomsu = Nomsu(nil, "[..]")
local line = Nomsu(self.source, "\n ") local line = Nomsu(nil, "\n ")
local _list_0 = self.value for _index_0 = 1, #self do
for _index_0 = 1, #_list_0 do local item = self[_index_0]
local item = _list_0[_index_0]
local item_nomsu = item:as_nomsu(true) local item_nomsu = item:as_nomsu(true)
if item_nomsu and #line + #", " + #item_nomsu <= MAX_LINE then if item_nomsu and #line + #", " + #item_nomsu <= MAX_LINE then
if #line.bits > 1 then if #line.bits > 1 then
@ -532,7 +558,7 @@ Tree("List", {
end end
if #line.bits > 1 then if #line.bits > 1 then
nomsu:append(line) nomsu:append(line)
line = Nomsu(item.source, "\n ") line = Nomsu(nil, "\n ")
end end
line:append(item_nomsu) line:append(item_nomsu)
end end
@ -544,11 +570,11 @@ Tree("List", {
end end
end end
}) })
Tree("Dict", { Tree("Dict", 'multi', {
as_lua = function(self, nomsu) as_lua = function(self, nomsu)
local lua = Lua.Value(self.source, "{") local lua = Lua.Value(nil, "{")
local line_length = 0 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) local entry_lua = entry:as_lua(nomsu)
lua:append(entry_lua) lua:append(entry_lua)
local entry_lua_str = tostring(entry_lua) local entry_lua_str = tostring(entry_lua)
@ -558,7 +584,7 @@ Tree("Dict", {
else else
line_length = line_length + #entry_lua_str line_length = line_length + #entry_lua_str
end end
if i < #self.value then if i < #self then
if line_length >= MAX_LINE then if line_length >= MAX_LINE then
lua:append(",\n ") lua:append(",\n ")
line_length = 0 line_length = 0
@ -576,8 +602,8 @@ Tree("Dict", {
inline = false inline = false
end end
if inline then if inline then
local nomsu = Nomsu(self.source, "{") local nomsu = Nomsu(nil, "{")
for i, entry in ipairs(self.value) do for i, entry in ipairs(self) do
local entry_nomsu = entry:as_nomsu(true) local entry_nomsu = entry:as_nomsu(true)
if not (entry_nomsu) then if not (entry_nomsu) then
return nil return nil
@ -594,11 +620,10 @@ Tree("Dict", {
if inline_version then if inline_version then
return inline_version return inline_version
end end
local nomsu = Nomsu(self.source, "{..}") local nomsu = Nomsu(nil, "{..}")
local line = Nomsu(self.source, "\n ") local line = Nomsu(nil, "\n ")
local _list_0 = self.value for _index_0 = 1, #self do
for _index_0 = 1, #_list_0 do local entry = self[_index_0]
local entry = _list_0[_index_0]
local entry_nomsu = entry:as_nomsu() local entry_nomsu = entry:as_nomsu()
if not (entry_nomsu) then if not (entry_nomsu) then
return nil return nil
@ -611,7 +636,7 @@ Tree("Dict", {
else else
if #line.bits > 1 then if #line.bits > 1 then
nomsu:append(line) nomsu:append(line)
line = Nomsu(bit.source, "\n ") line = Nomsu(nil, "\n ")
end end
line:append(entry_nomsu) line:append(entry_nomsu)
end end
@ -623,33 +648,31 @@ Tree("Dict", {
end end
end end
}) })
Tree("DictEntry", { Tree("DictEntry", 'multi', {
as_lua = function(self, nomsu) 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) local key_lua = key:as_lua(nomsu)
if not (key_lua.is_value) then if not (key_lua.is_value) then
local line, src = key.source:get_line(), key.source:get_text() error("Cannot use " .. tostring(colored.yellow(repr(key))) .. " as a dict key, since it's not an expression.", 0)
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(src)) .. " as a dict key, since it's not an expression.", 0)
end 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 if not (value_lua.is_value) then
local line, src = value.source:get_line(), value.source:get_text() error("Cannot use " .. tostring(colored.yellow(repr(value))) .. " as a dict value, since it's not an expression.", 0)
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(src)) .. " as a dict value, since it's not an expression.", 0)
end end
local key_str = tostring(key_lua):match([=[["']([a-zA-Z_][a-zA-Z0-9_]*)['"]]=]) local key_str = tostring(key_lua):match([=[["']([a-zA-Z_][a-zA-Z0-9_]*)['"]]=])
if key_str then 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 elseif tostring(key_lua):sub(1, 1) == "[" then
return Lua(key.source, "[ ", key_lua, "]=", value_lua) return Lua(nil, "[ ", key_lua, "]=", value_lua)
else else
return Lua(key.source, "[", key_lua, "]=", value_lua) return Lua(nil, "[", key_lua, "]=", value_lua)
end end
end, end,
as_nomsu = function(self, inline) as_nomsu = function(self, inline)
if inline == nil then if inline == nil then
inline = true inline = true
end end
local key, value = self.value[1], self.value[2] local key, value = self[1], self[2]
local key_nomsu = key:as_nomsu(true) local key_nomsu = key:as_nomsu(true)
if not (key_nomsu) then if not (key_nomsu) then
return nil return nil
@ -661,7 +684,7 @@ Tree("DictEntry", {
if value then if value then
value_nomsu = value:as_nomsu(true) value_nomsu = value:as_nomsu(true)
else else
value_nomsu = Nomsu(key.source, "") value_nomsu = Nomsu(nil, "")
end end
if inline and not value_nomsu then if inline and not value_nomsu then
return nil return nil
@ -675,43 +698,35 @@ Tree("DictEntry", {
return nil return nil
end end
end end
return Nomsu(key.source, key_nomsu, ":", value_nomsu) return Nomsu(nil, key_nomsu, ":", value_nomsu)
end end
}) })
Tree("IndexChain", { Tree("IndexChain", 'multi', {
as_lua = function(self, nomsu) 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 if not (lua.is_value) then
local line, src = self.value[1].source:get_line(), self.value[1].source:get_text() error("Cannot index " .. tostring(colored.yellow(repr(self[1]))) .. ", since it's not an expression.", 0)
error(tostring(line) .. ": Cannot index " .. tostring(colored.yellow(src)) .. ", since it's not an expression.", 0)
end end
local first_char = tostring(lua):sub(1, 1) local first_char = tostring(lua):sub(1, 1)
if first_char == "{" or first_char == '"' or first_char == "[" then if first_char == "{" or first_char == '"' or first_char == "[" then
lua:parenthesize() lua:parenthesize()
end end
for i = 2, #self.value do for i = 2, #self do
local _continue_0 = false local key = self[i]
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 = key:as_lua(nomsu) local key_lua = key:as_lua(nomsu)
if not (key_lua.is_value) then if not (key_lua.is_value) then
local line, src = key.source:get_line(), key.source:get_text() error("Cannot use " .. tostring(colored.yellow(repr(key))) .. " as an index, since it's not an expression.", 0)
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(src)) .. " as an index, since it's not an expression.", 0)
end end
if tostring(key_lua):sub(1, 1) == '[' then 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, " ]") lua:append("[ ", key_lua, " ]")
else else
lua:append("[", key_lua, "]") lua:append("[", key_lua, "]")
end end
_continue_0 = true
until true
if not _continue_0 then
break
end end
end end
return lua return lua
@ -720,8 +735,8 @@ Tree("IndexChain", {
if inline == nil then if inline == nil then
inline = false inline = false
end end
local nomsu = Nomsu(self.source) local nomsu = Nomsu()
for i, bit in ipairs(self.value) do for i, bit in ipairs(self) do
if i > 1 then if i > 1 then
nomsu:append(".") nomsu:append(".")
end end
@ -737,18 +752,18 @@ Tree("IndexChain", {
return nomsu return nomsu
end end
}) })
Tree("Number", { Tree("Number", 'single', {
as_lua = function(self, nomsu) as_lua = function(self, nomsu)
return Lua.Value(self.source, tostring(self.value)) return Lua.Value(nil, tostring(self.value))
end, end,
as_nomsu = function(self, inline) as_nomsu = function(self, inline)
if inline == nil then if inline == nil then
inline = false inline = false
end end
return Nomsu(self.source, tostring(self.value)) return Nomsu(nil, tostring(self.value))
end end
}) })
Tree("Var", { Tree("Var", 'single', {
as_lua_id = function(v) as_lua_id = function(v)
return "_" .. (v:gsub("%W", function(c) return "_" .. (v:gsub("%W", function(c)
if c == "_" then if c == "_" then
@ -759,16 +774,16 @@ Tree("Var", {
end)) end))
end, end,
as_lua = function(self, nomsu) 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, end,
as_nomsu = function(self, inline) as_nomsu = function(self, inline)
if inline == nil then if inline == nil then
inline = false inline = false
end end
return Nomsu(self.source, "%", self.value) return Nomsu(nil, "%", self.value)
end end
}) })
Tree("Word", { Tree("Word", 'single', {
as_lua = function(self, nomsu) as_lua = function(self, nomsu)
return error("Attempt to convert Word to lua") return error("Attempt to convert Word to lua")
end, end,
@ -776,12 +791,12 @@ Tree("Word", {
if inline == nil then if inline == nil then
inline = false inline = false
end end
return Nomsu(self.source, self.value) return Nomsu(nil, self.value)
end end
}) })
Tree("Comment", { Tree("Comment", 'single', {
as_lua = function(self, nomsu) 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, end,
as_nomsu = function(self, inline) as_nomsu = function(self, inline)
if inline == nil then if inline == nil then
@ -791,9 +806,9 @@ Tree("Comment", {
return nil return nil
end end
if self.value:match("\n") then if self.value:match("\n") then
return Nomsu(self.source, "#..", self.value:gsub("\n", "\n ")) return Nomsu(nil, "#..", self.value:gsub("\n", "\n "))
else else
return Nomsu(self.source, "#", self.value) return Nomsu(nil, "#", self.value)
end end
end end
}) })

View File

@ -13,46 +13,57 @@ Types.is_node = (n)->
type(n) == 'userdata' and getmetatable(n) and Types[n.type] == getmetatable(n) type(n) == 'userdata' and getmetatable(n) and Types[n.type] == getmetatable(n)
-- Helper method: -- Helper method:
Tree = (name, methods)-> Tree = (name, kind, methods)->
assert((kind == 'single') or (kind == 'multi'))
is_multi = (kind == 'multi')
with methods with methods
.__tostring = => "#{@name}(#{repr(@value)}, #{repr @source})" .with_value = (value)=> getmetatable(self)(value)
.with_value = (value)=> getmetatable(self)(value, @source)
.type = name .type = name
.name = name .name = name
.original_nomsu = => .is_multi = is_multi
leading_space = 0 if is_multi
src_file = FILE_CACHE[@source.filename] .__tostring = => "#{@name}(#{table.concat [repr(v) for v in *@], ', '})"
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
.map = (fn)=> .map = (fn)=>
if mapped = fn(self) if ret = fn(@)
return mapped return ret
if Tuple\is_instance(@value) new_vals = [v.map and v\map(fn) or v for v in *@]
return @with_value(Tuple(unpack([v.map and v\map(fn) or v for v in *@value]))) ret = getmetatable(self)(unpack(new_vals))
return self 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)=> 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)=> as_nomsu: (inline=false)=>
nomsu = @value\as_nomsu(true) nomsu = @value\as_nomsu(true)
if nomsu == nil and not inline if nomsu == nil and not inline
nomsu = @value\as_nomsu! nomsu = @value\as_nomsu!
return nomsu and Nomsu(@source, "\\:\n ", nomsu) return nomsu and Nomsu nil, "\\:\n ", nomsu
return nomsu and Nomsu(@source, "\\(", nomsu, ")") return nomsu and Nomsu nil, "\\(", nomsu, ")"
Tree "Block", map: (fn)=> fn(@) or @\map(fn)
Tree "Block", 'multi',
as_lua: (nomsu)=> as_lua: (nomsu)=>
lua = Lua(@source) lua = Lua!
for i,line in ipairs @value for i,line in ipairs @
line_lua = line\as_lua(nomsu) line_lua = line\as_lua(nomsu)
if i > 1 if i > 1
lua\append "\n" lua\append "\n"
@ -61,31 +72,31 @@ Tree "Block",
as_nomsu: (inline=false)=> as_nomsu: (inline=false)=>
if inline if inline
nomsu = Nomsu(@source) nomsu = Nomsu!
for i,line in ipairs @value for i,line in ipairs @
if i > 1 if i > 1
nomsu\append "; " nomsu\append "; "
line_nomsu = line\as_nomsu(true) line_nomsu = line\as_nomsu(true)
return nil unless line_nomsu return nil unless line_nomsu
nomsu\append line_nomsu nomsu\append line_nomsu
return nomsu return nomsu
nomsu = Nomsu(@source) nomsu = Nomsu!
for i, line in ipairs @value for i, line in ipairs @
line = assert(line\as_nomsu(nil, true), "Could not convert line to nomsu") line = assert(line\as_nomsu(nil, true), "Could not convert line to nomsu")
nomsu\append line nomsu\append line
if i < #@value if i < #@
nomsu\append "\n" nomsu\append "\n"
if tostring(line)\match("\n") if tostring(line)\match("\n")
nomsu\append "\n" nomsu\append "\n"
return nomsu return nomsu
math_expression = re.compile [[ ([+-] " ")* "%" (" " [*/^+-] (" " [+-])* " %")+ !. ]] math_expression = re.compile [[ ([+-] " ")* "%" (" " [*/^+-] (" " [+-])* " %")+ !. ]]
Tree "Action", Tree "Action", 'multi',
as_lua: (nomsu)=> as_lua: (nomsu)=>
stub = @get_stub! stub = @get_stub!
compile_action = nomsu.environment.COMPILE_ACTIONS[stub] compile_action = nomsu.environment.COMPILE_ACTIONS[stub]
if compile_action 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 -- Force all compile-time actions to take a tree location
args = [args[p-1] for p in *nomsu.environment.ARG_ORDERS[compile_action][stub]] args = [args[p-1] for p in *nomsu.environment.ARG_ORDERS[compile_action][stub]]
-- Force Lua to avoid tail call optimization for debugging purposes -- 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") if not ret then error("Failed to produce any Lua")
return ret return ret
action = rawget(nomsu.environment.ACTIONS, stub) action = rawget(nomsu.environment.ACTIONS, stub)
lua = Lua.Value(@source) lua = Lua.Value!
if not action and math_expression\match(stub) if not action and math_expression\match(stub)
-- This is a bit of a hack, but this code handles arbitrarily complex -- 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 -- math expressions like 2*x + 3^2 without having to define a single
-- action for every possibility. -- action for every possibility.
for i,tok in ipairs @value for i,tok in ipairs @
if tok.type == "Word" if tok.type == "Word"
lua\append tok.value lua\append tok.value
else else
tok_lua = tok\as_lua(nomsu) tok_lua = tok\as_lua(nomsu)
unless tok_lua.is_value unless tok_lua.is_value
src = tok.source\get_text! error("non-expression value inside math expression: #{colored.yellow repr(tok)}")
error("non-expression value inside math expression: #{colored.yellow src}")
if tok.type == "Action" if tok.type == "Action"
tok_lua\parenthesize! tok_lua\parenthesize!
lua\append tok_lua lua\append tok_lua
if i < #@value if i < #@
lua\append " " lua\append " "
return lua return lua
args = {} args = {}
for i, tok in ipairs @value for i, tok in ipairs @
if tok.type == "Word" then continue if tok.type == "Word" then continue
arg_lua = tok\as_lua(nomsu) arg_lua = tok\as_lua(nomsu)
unless arg_lua.is_value unless arg_lua.is_value
line, src = tok.source\get_line!, tok.source\get_text! 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
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 insert args, arg_lua
if action if action
@ -136,14 +145,14 @@ Tree "Action",
get_stub: (include_names=false)=> get_stub: (include_names=false)=>
bits = if include_names bits = if include_names
[(t.type == "Word" and t.value or "%#{t.value}") 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 *@value] else [(t.type == "Word" and t.value or "%") for t in *@]
return concat(bits, " ") return concat(bits, " ")
as_nomsu: (inline=false, can_use_colon=false)=> as_nomsu: (inline=false, can_use_colon=false)=>
if inline if inline
nomsu = Nomsu(@source) nomsu = Nomsu!
for i,bit in ipairs @value for i,bit in ipairs @
if bit.type == "Word" if bit.type == "Word"
if i > 1 if i > 1
nomsu\append " " nomsu\append " "
@ -158,11 +167,11 @@ Tree "Action",
nomsu\append arg_nomsu nomsu\append arg_nomsu
return nomsu return nomsu
else else
nomsu = Nomsu(@source) nomsu = Nomsu!
next_space = "" next_space = ""
-- TODO: track line length as we go and use 80-that instead of 80 for wrapping -- TODO: track line length as we go and use 80-that instead of 80 for wrapping
last_colon = nil last_colon = nil
for i,bit in ipairs @value for i,bit in ipairs @
if bit.type == "Word" if bit.type == "Word"
nomsu\append next_space, bit.value nomsu\append next_space, bit.value
next_space = " " next_space = " "
@ -189,9 +198,9 @@ Tree "Action",
-- These types carry their own indentation -- These types carry their own indentation
if bit.type != "List" and bit.type != "Dict" and bit.type != "Text" if bit.type != "List" and bit.type != "Dict" and bit.type != "Text"
if i == 1 if i == 1
arg_nomsu = Nomsu(bit.source, "(..)\n ", arg_nomsu) arg_nomsu = Nomsu(nil, "(..)\n ", arg_nomsu)
else 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") if last_colon == i-1 and (bit.type == "Action" or bit.type == "Block")
next_space = "" next_space = ""
@ -202,11 +211,11 @@ Tree "Action",
next_space = "\n.." next_space = "\n.."
return nomsu return nomsu
Tree "Text", Tree "Text", 'multi',
as_lua: (nomsu)=> as_lua: (nomsu)=>
lua = Lua.Value(@source) lua = Lua.Value!
string_buffer = "" string_buffer = ""
for bit in *@value for bit in *@
if type(bit) == "string" if type(bit) == "string"
string_buffer ..= bit string_buffer ..= bit
continue continue
@ -216,11 +225,10 @@ Tree "Text",
string_buffer = "" string_buffer = ""
bit_lua = bit\as_lua(nomsu) bit_lua = bit\as_lua(nomsu)
unless bit_lua.is_value unless bit_lua.is_value
line, src = bit.source\get_line!, bit.source\get_text! error "Cannot use #{colored.yellow repr(bit)} as a string interpolation value, since it's not an expression.", 0
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 #lua.bits > 0 then lua\append ".."
if bit.type != "Text" 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 lua\append bit_lua
if string_buffer ~= "" or #lua.bits == 0 if string_buffer ~= "" or #lua.bits == 0
@ -233,8 +241,8 @@ Tree "Text",
as_nomsu: (inline=false)=> as_nomsu: (inline=false)=>
if inline if inline
nomsu = Nomsu(@source, '"') nomsu = Nomsu(nil, '"')
for bit in *@value for bit in *@
if type(bit) == 'string' if type(bit) == 'string'
-- TODO: unescape better? -- TODO: unescape better?
nomsu\append (bit\gsub("\\","\\\\")\gsub("\n","\\n")) nomsu\append (bit\gsub("\\","\\\\")\gsub("\n","\\n"))
@ -251,8 +259,8 @@ Tree "Text",
inline_version = @as_nomsu(true) inline_version = @as_nomsu(true)
if inline_version and #inline_version <= MAX_LINE if inline_version and #inline_version <= MAX_LINE
return inline_version return inline_version
nomsu = Nomsu(@source, '".."\n ') nomsu = Nomsu(nil, '".."\n ')
for i, bit in ipairs @value for i, bit in ipairs @
if type(bit) == 'string' if type(bit) == 'string'
nomsu\append (bit\gsub("\\","\\\\")\gsub("\n","\n ")) nomsu\append (bit\gsub("\\","\\\\")\gsub("\n","\n "))
else else
@ -265,19 +273,18 @@ Tree "Text",
interp_nomsu = bit\as_nomsu! interp_nomsu = bit\as_nomsu!
return nil unless interp_nomsu return nil unless interp_nomsu
nomsu\append "\\\n ", interp_nomsu nomsu\append "\\\n ", interp_nomsu
if i < #@value if i < #@
nomsu\append "\n .." nomsu\append "\n .."
return nomsu return nomsu
Tree "List", Tree "List", 'multi',
as_lua: (nomsu)=> as_lua: (nomsu)=>
lua = Lua.Value @source, "{" lua = Lua.Value nil, "{"
line_length = 0 line_length = 0
for i, item in ipairs @value for i, item in ipairs @
item_lua = item\as_lua(nomsu) item_lua = item\as_lua(nomsu)
unless item_lua.is_value unless item_lua.is_value
line, src = item.source\get_line!, item.source\get_text! error "Cannot use #{colored.yellow repr(item)} as a list item, since it's not an expression.", 0
error "#{line}: Cannot use #{colored.yellow src} as a list item, since it's not an expression.", 0
lua\append item_lua lua\append item_lua
item_string = tostring(item_lua) item_string = tostring(item_lua)
last_line = item_string\match("[^\n]*$") last_line = item_string\match("[^\n]*$")
@ -285,7 +292,7 @@ Tree "List",
line_length = #last_line line_length = #last_line
else else
line_length += #last_line line_length += #last_line
if i < #@value if i < #@
if line_length >= MAX_LINE if line_length >= MAX_LINE
lua\append ",\n " lua\append ",\n "
line_length = 0 line_length = 0
@ -297,8 +304,8 @@ Tree "List",
as_nomsu: (inline=false)=> as_nomsu: (inline=false)=>
if inline if inline
nomsu = Nomsu(@source, "[") nomsu = Nomsu(nil, "[")
for i, item in ipairs @value for i, item in ipairs @
item_nomsu = item\as_nomsu(true) item_nomsu = item\as_nomsu(true)
return nil unless item_nomsu return nil unless item_nomsu
if i > 1 if i > 1
@ -310,9 +317,9 @@ Tree "List",
inline_version = @as_nomsu(true) inline_version = @as_nomsu(true)
if inline_version and #inline_version <= MAX_LINE if inline_version and #inline_version <= MAX_LINE
return inline_version return inline_version
nomsu = Nomsu(@source, "[..]") nomsu = Nomsu(nil, "[..]")
line = Nomsu(@source, "\n ") line = Nomsu(nil, "\n ")
for item in *@value for item in *@
item_nomsu = item\as_nomsu(true) item_nomsu = item\as_nomsu(true)
if item_nomsu and #line + #", " + #item_nomsu <= MAX_LINE if item_nomsu and #line + #", " + #item_nomsu <= MAX_LINE
if #line.bits > 1 if #line.bits > 1
@ -324,17 +331,17 @@ Tree "List",
return nil unless item_nomsu return nil unless item_nomsu
if #line.bits > 1 if #line.bits > 1
nomsu\append line nomsu\append line
line = Nomsu(item.source, "\n ") line = Nomsu(nil, "\n ")
line\append item_nomsu line\append item_nomsu
if #line.bits > 1 if #line.bits > 1
nomsu\append line nomsu\append line
return nomsu return nomsu
Tree "Dict", Tree "Dict", 'multi',
as_lua: (nomsu)=> as_lua: (nomsu)=>
lua = Lua.Value @source, "{" lua = Lua.Value nil, "{"
line_length = 0 line_length = 0
for i, entry in ipairs @value for i, entry in ipairs @
entry_lua = entry\as_lua(nomsu) entry_lua = entry\as_lua(nomsu)
lua\append entry_lua lua\append entry_lua
entry_lua_str = tostring(entry_lua) entry_lua_str = tostring(entry_lua)
@ -344,7 +351,7 @@ Tree "Dict",
line_length = #last_line line_length = #last_line
else else
line_length += #entry_lua_str line_length += #entry_lua_str
if i < #@value if i < #@
if line_length >= MAX_LINE if line_length >= MAX_LINE
lua\append ",\n " lua\append ",\n "
line_length = 0 line_length = 0
@ -356,8 +363,8 @@ Tree "Dict",
as_nomsu: (inline=false)=> as_nomsu: (inline=false)=>
if inline if inline
nomsu = Nomsu(@source, "{") nomsu = Nomsu(nil, "{")
for i, entry in ipairs @value for i, entry in ipairs @
entry_nomsu = entry\as_nomsu(true) entry_nomsu = entry\as_nomsu(true)
return nil unless entry_nomsu return nil unless entry_nomsu
if i > 1 if i > 1
@ -368,9 +375,9 @@ Tree "Dict",
else else
inline_version = @as_nomsu(true) inline_version = @as_nomsu(true)
if inline_version then return inline_version if inline_version then return inline_version
nomsu = Nomsu(@source, "{..}") nomsu = Nomsu(nil, "{..}")
line = Nomsu(@source, "\n ") line = Nomsu(nil, "\n ")
for entry in *@value for entry in *@
entry_nomsu = entry\as_nomsu! entry_nomsu = entry\as_nomsu!
return nil unless entry_nomsu return nil unless entry_nomsu
if #line + #tostring(entry_nomsu) <= MAX_LINE if #line + #tostring(entry_nomsu) <= MAX_LINE
@ -380,82 +387,78 @@ Tree "Dict",
else else
if #line.bits > 1 if #line.bits > 1
nomsu\append line nomsu\append line
line = Nomsu(bit.source, "\n ") line = Nomsu(nil, "\n ")
line\append entry_nomsu line\append entry_nomsu
if #line.bits > 1 if #line.bits > 1
nomsu\append line nomsu\append line
return nomsu return nomsu
Tree "DictEntry", Tree "DictEntry", 'multi',
as_lua: (nomsu)=> as_lua: (nomsu)=>
key, value = @value[1], @value[2] key, value = @[1], @[2]
key_lua = key\as_lua(nomsu) key_lua = key\as_lua(nomsu)
unless key_lua.is_value unless key_lua.is_value
line, src = key.source\get_line!, key.source\get_text! error "Cannot use #{colored.yellow repr(key)} as a dict key, since it's not an expression.", 0
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(nil, "true")
value_lua = value and value\as_lua(nomsu) or Lua.Value(key.source, "true")
unless value_lua.is_value unless value_lua.is_value
line, src = value.source\get_line!, value.source\get_text! error "Cannot use #{colored.yellow repr(value)} as a dict value, since it's not an expression.", 0
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_]*)['"]]=]) key_str = tostring(key_lua)\match([=[["']([a-zA-Z_][a-zA-Z0-9_]*)['"]]=])
return if key_str return if key_str
Lua key.source, key_str,"=",value_lua Lua nil, key_str,"=",value_lua
elseif tostring(key_lua)\sub(1,1) == "[" elseif tostring(key_lua)\sub(1,1) == "["
-- NOTE: this *must* use a space after the [ to avoid freaking out -- NOTE: this *must* use a space after the [ to avoid freaking out
-- Lua's parser if the inner expression is a long string. Lua -- Lua's parser if the inner expression is a long string. Lua
-- parses x[[[y]]] as x("[y]"), not as x["y"] -- parses x[[[y]]] as x("[y]"), not as x["y"]
Lua key.source, "[ ",key_lua,"]=",value_lua Lua nil, "[ ",key_lua,"]=",value_lua
else else
Lua key.source, "[",key_lua,"]=",value_lua Lua nil, "[",key_lua,"]=",value_lua
as_nomsu: (inline=true)=> as_nomsu: (inline=true)=>
key, value = @value[1], @value[2] key, value = @[1], @[2]
key_nomsu = key\as_nomsu(true) key_nomsu = key\as_nomsu(true)
return nil unless key_nomsu return nil unless key_nomsu
if key.type == "Action" or key.type == "Block" if key.type == "Action" or key.type == "Block"
key_nomsu\parenthesize! key_nomsu\parenthesize!
value_nomsu = if value value_nomsu = if value
value\as_nomsu(true) value\as_nomsu(true)
else Nomsu(key.source, "") else Nomsu(nil, "")
if inline and not value_nomsu then return nil if inline and not value_nomsu then return nil
if not value_nomsu if not value_nomsu
return nil if inline return nil if inline
value_nomsu = value\as_nomsu! value_nomsu = value\as_nomsu!
return nil unless value_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)=> as_lua: (nomsu)=>
lua = @value[1]\as_lua(nomsu) lua = @[1]\as_lua(nomsu)
unless lua.is_value unless lua.is_value
line, src = @value[1].source\get_line!, @value[1].source\get_text! error "Cannot index #{colored.yellow repr(@[1])}, since it's not an expression.", 0
error "#{line}: Cannot index #{colored.yellow src}, since it's not an expression.", 0
first_char = tostring(lua)\sub(1,1) first_char = tostring(lua)\sub(1,1)
if first_char == "{" or first_char == '"' or first_char == "[" if first_char == "{" or first_char == '"' or first_char == "["
lua\parenthesize! lua\parenthesize!
for i=2,#@value for i=2,#@
key = @value[i] key = @[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) key_lua = key\as_lua(nomsu)
unless key_lua.is_value unless key_lua.is_value
line, src = key.source\get_line!, key.source\get_text! error "Cannot use #{colored.yellow repr(key)} as an index, since it's not an expression.", 0
error "#{line}: Cannot use #{colored.yellow src} 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 -- NOTE: this *must* use a space after the [ to avoid freaking out
-- Lua's parser if the inner expression is a long string. Lua -- Lua's parser if the inner expression is a long string. Lua
-- parses x[[[y]]] as x("[y]"), not as x["y"] -- 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 else
lua\append "[",key_lua,"]" lua\append "[",key_lua,"]"
return lua return lua
as_nomsu: (inline=false)=> as_nomsu: (inline=false)=>
nomsu = Nomsu(@source) nomsu = Nomsu!
for i, bit in ipairs @value for i, bit in ipairs @
if i > 1 if i > 1
nomsu\append "." nomsu\append "."
bit_nomsu = bit\as_nomsu(true) bit_nomsu = bit\as_nomsu(true)
@ -465,39 +468,39 @@ Tree "IndexChain",
nomsu\append bit_nomsu nomsu\append bit_nomsu
return nomsu return nomsu
Tree "Number", Tree "Number", 'single',
as_lua: (nomsu)=> as_lua: (nomsu)=>
Lua.Value(@source, tostring(@value)) Lua.Value(nil, tostring(@value))
as_nomsu: (inline=false)=> as_nomsu: (inline=false)=>
return Nomsu(@source, tostring(@value)) return Nomsu(nil, tostring(@value))
Tree "Var", Tree "Var", 'single',
as_lua_id: (v)-> as_lua_id: (v)->
"_"..(v\gsub("%W", (c)-> if c == "_" then "__" else ("_%x")\format(c\byte!))) "_"..(v\gsub("%W", (c)-> if c == "_" then "__" else ("_%x")\format(c\byte!)))
as_lua: (nomsu)=> as_lua: (nomsu)=>
Lua.Value(@source, self.as_lua_id(@value)) Lua.Value(nil, self.as_lua_id(@value))
as_nomsu: (inline=false)=> as_nomsu: (inline=false)=>
return Nomsu(@source, "%", @value) return Nomsu(nil, "%", @value)
Tree "Word", Tree "Word", 'single',
as_lua: (nomsu)=> as_lua: (nomsu)=>
error("Attempt to convert Word to lua") error("Attempt to convert Word to lua")
as_nomsu: (inline=false)=> as_nomsu: (inline=false)=>
return Nomsu(@source, @value) return Nomsu(nil, @value)
Tree "Comment", Tree "Comment", 'single',
as_lua: (nomsu)=> as_lua: (nomsu)=>
Lua(@source, "--"..@value\gsub("\n","\n--").."\n") Lua(nil, "--"..@value\gsub("\n","\n--").."\n")
as_nomsu: (inline=false)=> as_nomsu: (inline=false)=>
return nil if inline return nil if inline
if @value\match("\n") if @value\match("\n")
return Nomsu(@source, "#..", @value\gsub("\n", "\n ")) return Nomsu(nil, "#..", @value\gsub("\n", "\n "))
else else
return Nomsu(@source, "#", @value) return Nomsu(nil, "#", @value)
return Types return Types

View File

@ -42,7 +42,7 @@ try: foo 99
assume ((\(5 + 5) as value) = 10) or barf "%tree as value failed." 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." assume ((repr [1,2]) = "{1, 2}") or barf "repr failed."