More correct and more working.
This commit is contained in:
parent
b05a46c78c
commit
222d4d0727
@ -5,14 +5,13 @@
|
|||||||
# Compile-time action to make compile-time actions:
|
# Compile-time action to make compile-time actions:
|
||||||
immediately:
|
immediately:
|
||||||
lua> ".."
|
lua> ".."
|
||||||
nomsu:define_compile_action("compile %actions to %lua", \(!! code location !!), function(\%actions, \%lua)
|
nomsu:define_compile_action("compile %actions to %lua", \(!! code location !!), function(tree, \%actions, \%lua)
|
||||||
local tree = nomsu.compilestack[#nomsu.compilestack];
|
|
||||||
local lua = Lua(tree.source, "nomsu:define_compile_action(");
|
local lua = Lua(tree.source, "nomsu:define_compile_action(");
|
||||||
local stubs = {};
|
local stubs = {};
|
||||||
for i, action in ipairs(\%actions.value) do
|
for i, action in ipairs(\%actions.value) do
|
||||||
stubs[i] = nomsu:tree_to_named_stub(action);
|
stubs[i] = nomsu:tree_to_named_stub(action);
|
||||||
end
|
end
|
||||||
lua:append(repr(stubs), ", ", repr(tree.source:get_line()), ", function(");
|
lua:append(repr(stubs), ", ", repr(tree.source:get_line()), ", function(tree");
|
||||||
local args = {};
|
local args = {};
|
||||||
for i,tok in ipairs(\%actions.value[1].value) do
|
for i,tok in ipairs(\%actions.value[1].value) do
|
||||||
if tok.type == "Var" then args[#args+1] = nomsu:var_to_lua_identifier(tok.value); end
|
if tok.type == "Var" then args[#args+1] = nomsu:var_to_lua_identifier(tok.value); end
|
||||||
@ -21,10 +20,10 @@ immediately:
|
|||||||
error("Invalid type for 'compile % to %', expected a dict with expr/statements, but got text.", 0);
|
error("Invalid type for 'compile % to %', expected a dict with expr/statements, but got text.", 0);
|
||||||
end
|
end
|
||||||
for i, arg in ipairs(args) do
|
for i, arg in ipairs(args) do
|
||||||
|
lua:append(", ");
|
||||||
lua:append(arg);
|
lua:append(arg);
|
||||||
if i < #args then lua:append(", ") end
|
|
||||||
end
|
end
|
||||||
local body_lua = nomsu:tree_to_lua(\%lua):as_statements();
|
local body_lua = nomsu:tree_to_lua(\%lua):as_statements("return ");
|
||||||
body_lua:declare_locals(args);
|
body_lua:declare_locals(args);
|
||||||
lua:append(")\\n ", body_lua, "\\nend);")
|
lua:append(")\\n ", body_lua, "\\nend);")
|
||||||
return lua;
|
return lua;
|
||||||
@ -34,7 +33,6 @@ immediately:
|
|||||||
immediately:
|
immediately:
|
||||||
compile [action %actions %body] to:
|
compile [action %actions %body] to:
|
||||||
lua> ".."
|
lua> ".."
|
||||||
local tree = nomsu.compilestack[#nomsu.compilestack];
|
|
||||||
local lua = Lua(tree.source, "nomsu:define_action(");
|
local lua = Lua(tree.source, "nomsu:define_action(");
|
||||||
local stubs = {};
|
local stubs = {};
|
||||||
for i, action in ipairs(\%actions.value) do
|
for i, action in ipairs(\%actions.value) do
|
||||||
@ -49,7 +47,7 @@ immediately:
|
|||||||
lua:append(arg);
|
lua:append(arg);
|
||||||
if i < #args then lua:append(", ") end
|
if i < #args then lua:append(", ") end
|
||||||
end
|
end
|
||||||
local body_lua = nomsu:tree_to_lua(\%body):as_statements();
|
local body_lua = nomsu:tree_to_lua(\%body):as_statements("return ");
|
||||||
body_lua:declare_locals(args);
|
body_lua:declare_locals(args);
|
||||||
lua:append(")\\n ", body_lua, "\\nend);")
|
lua:append(")\\n ", body_lua, "\\nend);")
|
||||||
return lua;
|
return lua;
|
||||||
@ -58,20 +56,19 @@ immediately:
|
|||||||
immediately:
|
immediately:
|
||||||
compile [parse %shorthand as %longhand] to:
|
compile [parse %shorthand as %longhand] to:
|
||||||
lua> ".."
|
lua> ".."
|
||||||
local tree = nomsu.compilestack[#nomsu.compilestack];
|
|
||||||
local lua = Lua(tree.source, "nomsu:define_compile_action(");
|
local lua = Lua(tree.source, "nomsu:define_compile_action(");
|
||||||
local stubs = {};
|
local stubs = {};
|
||||||
for i, action in ipairs(\%shorthand.value) do
|
for i, action in ipairs(\%shorthand.value) do
|
||||||
stubs[i] = nomsu:tree_to_named_stub(action);
|
stubs[i] = nomsu:tree_to_named_stub(action);
|
||||||
end
|
end
|
||||||
lua:append(repr(stubs), ", ", repr(tree.source:get_line()), ", function(");
|
lua:append(repr(stubs), ", ", repr(tree.source:get_line()), ", function(tree");
|
||||||
local args = {};
|
local args = {};
|
||||||
for i,tok in ipairs(\%shorthand.value[1].value) do
|
for i,tok in ipairs(\%shorthand.value[1].value) do
|
||||||
if tok.type == "Var" then args[#args+1] = nomsu:var_to_lua_identifier(tok.value); end
|
if tok.type == "Var" then args[#args+1] = nomsu:var_to_lua_identifier(tok.value); end
|
||||||
end
|
end
|
||||||
for i, arg in ipairs(args) do
|
for i, arg in ipairs(args) do
|
||||||
|
lua:append(", ");
|
||||||
lua:append(arg);
|
lua:append(arg);
|
||||||
if i < #args then lua:append(", ") end
|
|
||||||
end
|
end
|
||||||
local template;
|
local template;
|
||||||
if \%longhand.type == "Block" then
|
if \%longhand.type == "Block" then
|
||||||
@ -127,14 +124,14 @@ immediately:
|
|||||||
=lua "nomsu:tree_to_stub(\%tree)"
|
=lua "nomsu:tree_to_stub(\%tree)"
|
||||||
|
|
||||||
immediately:
|
immediately:
|
||||||
compile [%tree's source code, %tree' source code] to {expr:"\(%tree as lua expr).source:get_text()"}
|
compile [%tree's source code, %tree' source code] to: LuaValue "\(%tree as lua expr).source:get_text()"
|
||||||
|
|
||||||
compile [repr %obj] to {expr:"repr(\(%obj as lua expr))"}
|
compile [repr %obj] to: LuaValue "repr(\(%obj as lua expr))"
|
||||||
compile [type of %obj] to {expr:"type(\(%obj as lua expr))"}
|
compile [type of %obj] to: LuaValue "type(\(%obj as lua expr))"
|
||||||
|
|
||||||
immediately:
|
immediately:
|
||||||
compile [nomsu] to {expr:"nomsu"}
|
compile [nomsu] to: LuaValue "nomsu"
|
||||||
compile [%var as lua identifier] to {expr:"nomsu:var_to_lua_identifier(\(%var as lua expr))"}
|
compile [%var as lua identifier] to: LuaValue "nomsu:var_to_lua_identifier(\(%var as lua expr))"
|
||||||
|
|
||||||
action [action %names metadata]:
|
action [action %names metadata]:
|
||||||
=lua "nomsu.action_metadata[ACTIONS[\%names]]"
|
=lua "nomsu.action_metadata[ACTIONS[\%names]]"
|
||||||
@ -151,9 +148,8 @@ action [help %action]:
|
|||||||
|
|
||||||
# Compiler tools
|
# Compiler tools
|
||||||
immediately:
|
immediately:
|
||||||
#local def_metadata = nomsu.tree_metadata[nomsu.compilestack[#nomsu.compilestack]];
|
compile [run %code] to:
|
||||||
compile [run %code] to {..}
|
LuaValue ".."
|
||||||
expr: ".."
|
|
||||||
nomsu:run(\(%code as lua expr), '\
|
nomsu:run(\(%code as lua expr), '\
|
||||||
=lua "nomsu:get_line_number(nomsu.compilestack[#nomsu.compilestack])"
|
=lua "nomsu:get_line_number(nomsu.compilestack[#nomsu.compilestack])"
|
||||||
..')
|
..')
|
||||||
@ -164,44 +160,40 @@ immediately:
|
|||||||
compile [show lua %block] to:
|
compile [show lua %block] to:
|
||||||
lua> ".."
|
lua> ".."
|
||||||
local \%lua = nomsu:tree_to_lua(\%block);
|
local \%lua = nomsu:tree_to_lua(\%block);
|
||||||
return {statements = "print("..repr(\%lua.statements or \%lua.expr)..");"};
|
return Lua(\%block.source, "print(", repr(tostring(\%lua)), ");");
|
||||||
|
|
||||||
immediately:
|
immediately:
|
||||||
compile [say %message] to:
|
compile [say %message] to:
|
||||||
lua> ".."
|
lua> ".."
|
||||||
|
local tree = nomsu.compilestack[#nomsu.compilestack];
|
||||||
if \%message.type == "Text" then
|
if \%message.type == "Text" then
|
||||||
return {statements="print("..\(%message as lua expr)..");"};
|
return Lua(tree.source, "print(", \(%message as lua expr), ");");
|
||||||
else
|
else
|
||||||
return {statements="print(stringify("..\(%message as lua expr).."));"};
|
return Lua(tree.source, "print(stringify(", \(%message as lua expr), "));");
|
||||||
end
|
end
|
||||||
|
|
||||||
# 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
|
||||||
put code after a return statement, unless you wrap it in a block.
|
put code after a return statement, unless you wrap it in a block.
|
||||||
compile [return] to {statements:"do return; end"}
|
compile [return] to: Lua "do return; end"
|
||||||
compile [return %return_value] to {statements:"do return \(%return_value as lua expr); end"}
|
compile [return %return_value] to: Lua "do return \(%return_value as lua expr); end"
|
||||||
|
|
||||||
# Error functions
|
# Error functions
|
||||||
immediately:
|
immediately:
|
||||||
compile [barf] to {statements:"error(nil, 0);"}
|
compile [barf] to: LuaValue "error(nil, 0);"
|
||||||
compile [barf %msg] to {statements:"error(\(%msg as lua expr), 0);"}
|
compile [barf %msg] to: LuaValue "error(\(%msg as lua expr), 0);"
|
||||||
compile [assume %condition] to:
|
compile [assume %condition] to:
|
||||||
lua> "local \%assumption = 'Assumption failed: '..\%condition.source:get_text();"
|
lua> "local \%assumption = 'Assumption failed: '..\%condition.source:get_text();"
|
||||||
return {..}
|
return:
|
||||||
statements:".."
|
Lua ".."
|
||||||
if not \(%condition as lua expr) then
|
if not \(%condition as lua expr) then
|
||||||
error(\(repr %assumption), 0);
|
error(\(repr %assumption), 0);
|
||||||
end
|
end
|
||||||
compile [assume %condition or barf %msg] to {..}
|
|
||||||
statements:".."
|
|
||||||
if not \(%condition as lua expr) then
|
|
||||||
error(\(%msg as lua expr), 0);
|
|
||||||
end
|
|
||||||
|
|
||||||
# Literals
|
# Literals
|
||||||
immediately:
|
immediately:
|
||||||
compile [yes] to {expr:"true"}
|
compile [yes] to: LuaValue "true"
|
||||||
compile [no] to {expr:"false"}
|
compile [no] to: LuaValue "false"
|
||||||
compile [nothing, nil, null] to {expr:"nil"}
|
compile [nothing, nil, null] to: LuaValue "nil"
|
||||||
|
|
||||||
|
40
lua_obj.moon
40
lua_obj.moon
@ -3,29 +3,26 @@ immutable = require 'immutable'
|
|||||||
local Lua, LuaValue, Location
|
local Lua, LuaValue, Location
|
||||||
export LINE_STARTS
|
export LINE_STARTS
|
||||||
|
|
||||||
Location = immutable {"text_name","text","start","stop"}, {
|
Location = immutable {"filename","start","stop"}, {
|
||||||
name:"Location"
|
name:"Location"
|
||||||
__new: (text_name, text, start, stop)=>
|
__new: (filename, start, stop)=>
|
||||||
assert(type(text_name) == 'string')
|
assert(type(filename) == 'string' and type(start) == 'number' and type(stop) == 'number')
|
||||||
assert(type(text) == 'string')
|
return filename, start, stop or start
|
||||||
assert(type(start) == 'number')
|
__tostring: => "Location(\"#{@filename}\", #{@start}, #{@stop})"
|
||||||
assert(type(stop or start) == 'number')
|
|
||||||
return text_name, text, start, stop or start
|
|
||||||
__tostring: => "#{@text_name}[#{@start}:#{@stop}]"
|
|
||||||
__lt: (other)=>
|
__lt: (other)=>
|
||||||
assert(@text == other.text, "Cannot compare sources from different texts")
|
assert(@filename == other.filename, "Cannot compare sources from different files")
|
||||||
return if @start == other.start
|
return if @start == other.start
|
||||||
@stop < other.stop
|
@stop < other.stop
|
||||||
else @start < other.start
|
else @start < other.start
|
||||||
__le: (other)=>
|
__le: (other)=>
|
||||||
assert(@text == other.text, "Cannot compare sources from different texts")
|
assert(@filename == other.filename, "Cannot compare sources from different files")
|
||||||
return if @start == other.start
|
return if @start == other.start
|
||||||
@stop <= other.stop
|
@stop <= other.stop
|
||||||
else @start <= other.start
|
else @start <= other.start
|
||||||
get_text: => @text\sub(@start,@stop)
|
get_text: => FILE_CACHE[@filename]\sub(@start,@stop)
|
||||||
get_line_number: =>
|
get_line_number: =>
|
||||||
-- TODO: do a binary search if this is actually slow, which I doubt
|
-- TODO: do a binary search if this is actually slow, which I doubt
|
||||||
line_starts = LINE_STARTS[@text]
|
line_starts = LINE_STARTS[FILE_CACHE[@filename]]
|
||||||
start_line = 1
|
start_line = 1
|
||||||
while (line_starts[start_line+1] or (#src+1)) <= @start
|
while (line_starts[start_line+1] or (#src+1)) <= @start
|
||||||
start_line += 1
|
start_line += 1
|
||||||
@ -33,12 +30,12 @@ Location = immutable {"text_name","text","start","stop"}, {
|
|||||||
while (line_starts[stop_line+1] or (#src+1)) <= @stop
|
while (line_starts[stop_line+1] or (#src+1)) <= @stop
|
||||||
stop_line += 1
|
stop_line += 1
|
||||||
return start_line, stop_line
|
return start_line, stop_line
|
||||||
get_line: => "#{@text_name}:#{@get_line_number!}"
|
get_line: => "#{@filename}:#{@get_line_number!}"
|
||||||
get_line_range: =>
|
get_line_range: =>
|
||||||
start_line, stop_line = @get_line_number!
|
start_line, stop_line = @get_line_number!
|
||||||
return if stop_line == start_line
|
return if stop_line == start_line
|
||||||
"#{text_name}:#{start_line}"
|
"#{@filename}:#{start_line}"
|
||||||
else "#{text_name}:#{start_line}-#{stop_line}"
|
else "#{@filename}:#{start_line}-#{stop_line}"
|
||||||
}
|
}
|
||||||
|
|
||||||
class Lua
|
class Lua
|
||||||
@ -46,6 +43,9 @@ class Lua
|
|||||||
is_value: false
|
is_value: false
|
||||||
|
|
||||||
new: (@source, ...)=>
|
new: (@source, ...)=>
|
||||||
|
if type(@source) == 'string'
|
||||||
|
filename,start,stop = @source\match("^(.-)[(%d+):(%d+)]$")
|
||||||
|
@source = Location(filename, tonumber(start), tonumber(stop))
|
||||||
for i=1,select("#",...)
|
for i=1,select("#",...)
|
||||||
x = select(i,...)
|
x = select(i,...)
|
||||||
assert(type(x) != 'table' or getmetatable(x))
|
assert(type(x) != 'table' or getmetatable(x))
|
||||||
@ -109,7 +109,7 @@ class Lua
|
|||||||
-- Return a mapping from output (lua) character number to input (nomsu) character number
|
-- Return a mapping from output (lua) character number to input (nomsu) character number
|
||||||
lua_str = tostring(self)
|
lua_str = tostring(self)
|
||||||
metadata = {
|
metadata = {
|
||||||
nomsu_filename:@source.text_name, nomsu_file:@source.text,
|
nomsu_filename:@source.filename
|
||||||
lua_filename:lua_chunkname, lua_file:lua_str
|
lua_filename:lua_chunkname, lua_file:lua_str
|
||||||
lua_to_nomsu: {}, nomsu_to_lua: {}
|
lua_to_nomsu: {}, nomsu_to_lua: {}
|
||||||
}
|
}
|
||||||
@ -122,7 +122,7 @@ class Lua
|
|||||||
for b in *lua.bits
|
for b in *lua.bits
|
||||||
walk b
|
walk b
|
||||||
lua_stop = lua_offset
|
lua_stop = lua_offset
|
||||||
nomsu_src, lua_src = lua.source, Location(lua_chunkname, lua_str, lua_start, lua_stop)
|
nomsu_src, lua_src = lua.source, Location(lua_chunkname, lua_start, lua_stop)
|
||||||
metadata.lua_to_nomsu[lua_src] = nomsu_src
|
metadata.lua_to_nomsu[lua_src] = nomsu_src
|
||||||
metadata.nomsu_to_lua[nomsu_src] = lua_src
|
metadata.nomsu_to_lua[nomsu_src] = lua_src
|
||||||
walk self
|
walk self
|
||||||
@ -143,9 +143,9 @@ class LuaValue extends Lua
|
|||||||
assert(not ret\match(".*table: 0x.*"))
|
assert(not ret\match(".*table: 0x.*"))
|
||||||
return ret
|
return ret
|
||||||
|
|
||||||
as_statements: =>
|
as_statements: (prefix="", suffix=";")=>
|
||||||
bits = {unpack @bits}
|
bits = {prefix, unpack @bits}
|
||||||
bits[#bits+1] = ";"
|
bits[#bits+1] = suffix
|
||||||
return Lua(@source, unpack(bits))
|
return Lua(@source, unpack(bits))
|
||||||
|
|
||||||
parenthesize: =>
|
parenthesize: =>
|
||||||
|
95
nomsu.lua
95
nomsu.lua
@ -32,7 +32,7 @@ do
|
|||||||
local _obj_0 = require("lua_obj")
|
local _obj_0 = require("lua_obj")
|
||||||
Lua, LuaValue, Location = _obj_0.Lua, _obj_0.LuaValue, _obj_0.Location
|
Lua, LuaValue, Location = _obj_0.Lua, _obj_0.LuaValue, _obj_0.Location
|
||||||
end
|
end
|
||||||
local FILE_CACHE = setmetatable({ }, {
|
FILE_CACHE = setmetatable({ }, {
|
||||||
__index = function(self, filename)
|
__index = function(self, filename)
|
||||||
local file = io.open(filename)
|
local file = io.open(filename)
|
||||||
if not (file) then
|
if not (file) then
|
||||||
@ -196,7 +196,7 @@ do
|
|||||||
pos = pos + #lpeg.userdata.source_code:match("[ \t\n\r]*", pos)
|
pos = pos + #lpeg.userdata.source_code:match("[ \t\n\r]*", pos)
|
||||||
end
|
end
|
||||||
local line_no = 1
|
local line_no = 1
|
||||||
local text_loc = Location(lpeg.userdata.filename, src, pos)
|
local text_loc = Location(lpeg.userdata.filename, pos)
|
||||||
line_no = text_loc:get_line_number()
|
line_no = text_loc:get_line_number()
|
||||||
local prev_line = src:sub(LINE_STARTS[src][line_no - 1] or 1, LINE_STARTS[src][line_no] - 1)
|
local prev_line = src:sub(LINE_STARTS[src][line_no - 1] or 1, LINE_STARTS[src][line_no] - 1)
|
||||||
local err_line = src:sub(LINE_STARTS[src][line_no], (LINE_STARTS[src][line_no + 1] or 0) - 1)
|
local err_line = src:sub(LINE_STARTS[src][line_no], (LINE_STARTS[src][line_no + 1] or 0) - 1)
|
||||||
@ -215,7 +215,7 @@ setmetatable(NOMSU_DEFS, {
|
|||||||
if type(value) == 'table' then
|
if type(value) == 'table' then
|
||||||
error("Not a tuple: " .. tostring(repr(value)))
|
error("Not a tuple: " .. tostring(repr(value)))
|
||||||
end
|
end
|
||||||
local loc = Location(lpeg.userdata.filename, lpeg.userdata.source_code, start, stop)
|
local loc = Location(lpeg.userdata.filename, start, stop)
|
||||||
local node = Types[key](value, loc)
|
local node = Types[key](value, loc)
|
||||||
return node
|
return node
|
||||||
end
|
end
|
||||||
@ -282,9 +282,6 @@ do
|
|||||||
end
|
end
|
||||||
arg_positions = _accum_0
|
arg_positions = _accum_0
|
||||||
end
|
end
|
||||||
if #arg_positions ~= #args then
|
|
||||||
error("Mismatch in args between lua function's " .. tostring(repr(fn_arg_positions)) .. " and stub's " .. tostring(repr(args)) .. " for " .. tostring(repr(stub)), 0)
|
|
||||||
end
|
|
||||||
arg_orders[stub] = arg_positions
|
arg_orders[stub] = arg_positions
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@ -379,7 +376,8 @@ do
|
|||||||
local tree = self:parse(nomsu_code, filename)
|
local tree = self:parse(nomsu_code, filename)
|
||||||
assert(tree, "Failed to parse: " .. tostring(nomsu_code))
|
assert(tree, "Failed to parse: " .. tostring(nomsu_code))
|
||||||
assert(tree.type == "File", "Attempt to run non-file: " .. tostring(tree.type))
|
assert(tree.type == "File", "Attempt to run non-file: " .. tostring(tree.type))
|
||||||
local lua = self:tree_to_lua(tree):as_statements():with_locals_declared()
|
local lua = self:tree_to_lua(tree):as_statements()
|
||||||
|
lua:declare_locals()
|
||||||
lua:prepend("-- File: " .. tostring(filename) .. "\n")
|
lua:prepend("-- File: " .. tostring(filename) .. "\n")
|
||||||
return self:run_lua(lua, filename .. ".lua")
|
return self:run_lua(lua, filename .. ".lua")
|
||||||
end,
|
end,
|
||||||
@ -449,8 +447,9 @@ do
|
|||||||
_chunk_counter = _chunk_counter + 1
|
_chunk_counter = _chunk_counter + 1
|
||||||
filename = "<lua chunk #" .. tostring(_chunk_counter) .. ">"
|
filename = "<lua chunk #" .. tostring(_chunk_counter) .. ">"
|
||||||
else
|
else
|
||||||
filename = lua.source.text_name .. ".lua"
|
filename = lua.source.filename .. ".lua"
|
||||||
end
|
end
|
||||||
|
FILE_CACHE[filename] = tostring(lua)
|
||||||
end
|
end
|
||||||
if type(lua) ~= 'string' then
|
if type(lua) ~= 'string' then
|
||||||
local metadata
|
local metadata
|
||||||
@ -475,7 +474,7 @@ do
|
|||||||
return tree.value[1]
|
return tree.value[1]
|
||||||
end
|
end
|
||||||
local lua = Lua(tree.source, "return ", self:tree_to_lua(tree), ";")
|
local lua = Lua(tree.source, "return ", self:tree_to_lua(tree), ";")
|
||||||
return self:run_lua(lua, tree.source.text_name)
|
return self:run_lua(lua, tree.source.filename)
|
||||||
end,
|
end,
|
||||||
tree_to_nomsu = function(self, tree, indentation, max_line, expr_type)
|
tree_to_nomsu = function(self, tree, indentation, max_line, expr_type)
|
||||||
if indentation == nil then
|
if indentation == nil then
|
||||||
@ -896,9 +895,9 @@ do
|
|||||||
file_lua:declare_locals()
|
file_lua:declare_locals()
|
||||||
return file_lua
|
return file_lua
|
||||||
elseif "Comment" == _exp_0 then
|
elseif "Comment" == _exp_0 then
|
||||||
return Lua(tree.source, "--" .. tree.value:gsub("\n", "\n--"))
|
return Lua(tree.source, "--" .. tree.value:gsub("\n", "\n--") .. "\n")
|
||||||
elseif "Nomsu" == _exp_0 then
|
elseif "Nomsu" == _exp_0 then
|
||||||
return Lua(tree.source, "nomsu:parse(", tree.source:get_text(), ", ", repr(tree.source.text_name), ")")
|
return Lua(tree.source, "nomsu:parse(", tree.source:get_text(), ", ", repr(tree.source.filename), ")")
|
||||||
elseif "Block" == _exp_0 then
|
elseif "Block" == _exp_0 then
|
||||||
local block_lua = Lua(tree.source)
|
local block_lua = Lua(tree.source)
|
||||||
for i, arg in ipairs(tree.value) do
|
for i, arg in ipairs(tree.value) do
|
||||||
@ -941,14 +940,14 @@ do
|
|||||||
local _list_1 = metadata.arg_orders[stub]
|
local _list_1 = metadata.arg_orders[stub]
|
||||||
for _index_0 = 1, #_list_1 do
|
for _index_0 = 1, #_list_1 do
|
||||||
local p = _list_1[_index_0]
|
local p = _list_1[_index_0]
|
||||||
_accum_0[_len_0] = args[p]
|
_accum_0[_len_0] = args[p - 1]
|
||||||
_len_0 = _len_0 + 1
|
_len_0 = _len_0 + 1
|
||||||
end
|
end
|
||||||
new_args = _accum_0
|
new_args = _accum_0
|
||||||
end
|
end
|
||||||
args = new_args
|
args = new_args
|
||||||
end
|
end
|
||||||
local lua = action(unpack(args))
|
local lua = action(tree, unpack(args))
|
||||||
remove(self.compilestack)
|
remove(self.compilestack)
|
||||||
return lua
|
return lua
|
||||||
elseif not metadata and self.__class.math_patt:match(stub) then
|
elseif not metadata and self.__class.math_patt:match(stub) then
|
||||||
@ -1367,15 +1366,61 @@ do
|
|||||||
return "nomsu.moon:" .. tostring(debug_getinfo(2).currentline)
|
return "nomsu.moon:" .. tostring(debug_getinfo(2).currentline)
|
||||||
end
|
end
|
||||||
local nomsu = self
|
local nomsu = self
|
||||||
self:define_compile_action("immediately %block", get_line_no(), function(_block)
|
self:define_compile_action("immediately %block", get_line_no(), function(self, _block)
|
||||||
local lua = nomsu:tree_to_lua(_block):as_statements()
|
local lua = nomsu:tree_to_lua(_block):as_statements()
|
||||||
lua:declare_locals()
|
lua:declare_locals()
|
||||||
nomsu:run_lua(lua)
|
nomsu:run_lua(lua)
|
||||||
return Lua(_block.source, "if IMMEDIATE then\n", lua, "\nend")
|
return Lua(self.source, "if IMMEDIATE then\n", lua, "\nend")
|
||||||
end)
|
end)
|
||||||
self:define_compile_action("lua> %code", get_line_no(), function(_code)
|
self:define_compile_action("Lua %code", get_line_no(), function(self, _code)
|
||||||
if _code.type ~= "Text" then
|
if _code.type ~= "Text" then
|
||||||
return LuaValue(_code.source, "nomsu:run_lua(", nomsu:tree_to_lua(_code), ")")
|
return LuaValue(self.source, "Lua(", repr(_code.source), ", ", nomsu:tree_to_lua(_code), ")")
|
||||||
|
end
|
||||||
|
local lua = LuaValue(self.source, "Lua(", repr(_code.source))
|
||||||
|
local _list_1 = _code.value
|
||||||
|
for _index_0 = 1, #_list_1 do
|
||||||
|
local bit = _list_1[_index_0]
|
||||||
|
lua:append(", ")
|
||||||
|
if type(bit) == "string" then
|
||||||
|
lua:append(repr(bit))
|
||||||
|
else
|
||||||
|
local bit_lua = nomsu:tree_to_lua(bit)
|
||||||
|
if not (bit_lua.is_value) then
|
||||||
|
local line, src = bit.source:get_line(), bit.source:get_text()
|
||||||
|
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(src)) .. " as a string interpolation value, since it's not an expression.", 0)
|
||||||
|
end
|
||||||
|
lua:append(bit_lua)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
lua:append(")")
|
||||||
|
return lua
|
||||||
|
end)
|
||||||
|
self:define_compile_action("LuaValue %code", get_line_no(), function(self, _code)
|
||||||
|
if _code.type ~= "Text" then
|
||||||
|
return LuaValue(self.source, "LuaValue(", repr(_code.source), ", ", nomsu:tree_to_lua(_code), ")")
|
||||||
|
end
|
||||||
|
local lua = LuaValue(self.source, "LuaValue(", repr(_code.source))
|
||||||
|
local _list_1 = _code.value
|
||||||
|
for _index_0 = 1, #_list_1 do
|
||||||
|
local bit = _list_1[_index_0]
|
||||||
|
lua:append(", ")
|
||||||
|
if type(bit) == "string" then
|
||||||
|
lua:append(repr(bit))
|
||||||
|
else
|
||||||
|
local bit_lua = nomsu:tree_to_lua(bit)
|
||||||
|
if not (bit_lua.is_value) then
|
||||||
|
local line, src = bit.source:get_line(), bit.source:get_text()
|
||||||
|
error(tostring(line) .. ": Cannot use " .. tostring(colored.yellow(src)) .. " as a string interpolation value, since it's not an expression.", 0)
|
||||||
|
end
|
||||||
|
lua:append(bit_lua)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
lua:append(")")
|
||||||
|
return lua
|
||||||
|
end)
|
||||||
|
self:define_compile_action("lua> %code", get_line_no(), function(self, _code)
|
||||||
|
if _code.type ~= "Text" then
|
||||||
|
return LuaValue(self.source, "nomsu:run_lua(", nomsu:tree_to_lua(_code), ")")
|
||||||
end
|
end
|
||||||
local lua = Lua(_code.source)
|
local lua = Lua(_code.source)
|
||||||
local _list_1 = _code.value
|
local _list_1 = _code.value
|
||||||
@ -1394,11 +1439,11 @@ do
|
|||||||
end
|
end
|
||||||
return lua
|
return lua
|
||||||
end)
|
end)
|
||||||
self:define_compile_action("=lua %code", get_line_no(), function(_code)
|
self:define_compile_action("=lua %code", get_line_no(), function(self, _code)
|
||||||
if _code.type ~= "Text" then
|
if _code.type ~= "Text" then
|
||||||
return LuaValue(_code.source, "nomsu:run_lua(", nomsu:tree_to_lua(_code), ")")
|
return LuaValue(self.source, "nomsu:run_lua(", nomsu:tree_to_lua(_code), ")")
|
||||||
end
|
end
|
||||||
local lua = LuaValue(_code.source)
|
local lua = LuaValue(self.source)
|
||||||
local _list_1 = _code.value
|
local _list_1 = _code.value
|
||||||
for _index_0 = 1, #_list_1 do
|
for _index_0 = 1, #_list_1 do
|
||||||
local bit = _list_1[_index_0]
|
local bit = _list_1[_index_0]
|
||||||
@ -1415,18 +1460,16 @@ do
|
|||||||
end
|
end
|
||||||
return lua
|
return lua
|
||||||
end)
|
end)
|
||||||
self:define_compile_action("!! code location !!", get_line_no(), function()
|
self:define_compile_action("!! code location !!", get_line_no(), function(self)
|
||||||
local tree = nomsu.compilestack[#nomsu.compilestack - 1]
|
return LuaValue(self.source, repr(self.source))
|
||||||
return LuaValue(tree.source, repr(tostring(tree.source)))
|
|
||||||
end)
|
end)
|
||||||
self:define_action("run file %filename", get_line_no(), function(_filename)
|
self:define_action("run file %filename", get_line_no(), function(_filename)
|
||||||
return nomsu:run_file(_filename)
|
return nomsu:run_file(_filename)
|
||||||
end)
|
end)
|
||||||
return self:define_compile_action("use %filename", get_line_no(), function(_filename)
|
return self:define_compile_action("use %filename", get_line_no(), function(self, _filename)
|
||||||
local tree = nomsu.compilestack[#nomsu.compilestack - 1]
|
|
||||||
local filename = nomsu:tree_to_value(_filename)
|
local filename = nomsu:tree_to_value(_filename)
|
||||||
nomsu:use_file(filename)
|
nomsu:use_file(filename)
|
||||||
return LuaValue(tree.source, "nomsu:use_file(" .. tostring(repr(filename)) .. ")")
|
return LuaValue(self.source, "nomsu:use_file(" .. tostring(repr(filename)) .. ")")
|
||||||
end)
|
end)
|
||||||
end
|
end
|
||||||
}
|
}
|
||||||
|
85
nomsu.moon
85
nomsu.moon
@ -36,6 +36,7 @@ debug_getinfo = debug.getinfo
|
|||||||
-- Add a ((%x foo %y) where {x:"asdf", y:"fdsa"}) compile-time action for substitution
|
-- Add a ((%x foo %y) where {x:"asdf", y:"fdsa"}) compile-time action for substitution
|
||||||
-- Allow plain text backslash like: "\n" in longstrings without requiring "\\n"
|
-- Allow plain text backslash like: "\n" in longstrings without requiring "\\n"
|
||||||
|
|
||||||
|
export FILE_CACHE
|
||||||
FILE_CACHE = setmetatable {}, {
|
FILE_CACHE = setmetatable {}, {
|
||||||
__index: (filename)=>
|
__index: (filename)=>
|
||||||
file = io.open(filename)
|
file = io.open(filename)
|
||||||
@ -149,7 +150,7 @@ NOMSU_DEFS = with {}
|
|||||||
if lpeg.userdata.source_code\sub(pos,pos)\match("[\r\n]")
|
if lpeg.userdata.source_code\sub(pos,pos)\match("[\r\n]")
|
||||||
pos += #lpeg.userdata.source_code\match("[ \t\n\r]*", pos)
|
pos += #lpeg.userdata.source_code\match("[ \t\n\r]*", pos)
|
||||||
line_no = 1
|
line_no = 1
|
||||||
text_loc = Location(lpeg.userdata.filename, src, pos)
|
text_loc = Location(lpeg.userdata.filename, pos)
|
||||||
line_no = text_loc\get_line_number!
|
line_no = text_loc\get_line_number!
|
||||||
prev_line = src\sub(LINE_STARTS[src][line_no-1] or 1, LINE_STARTS[src][line_no]-1)
|
prev_line = src\sub(LINE_STARTS[src][line_no-1] or 1, LINE_STARTS[src][line_no]-1)
|
||||||
err_line = src\sub(LINE_STARTS[src][line_no], (LINE_STARTS[src][line_no+1] or 0)-1)
|
err_line = src\sub(LINE_STARTS[src][line_no], (LINE_STARTS[src][line_no+1] or 0)-1)
|
||||||
@ -162,7 +163,7 @@ NOMSU_DEFS = with {}
|
|||||||
setmetatable(NOMSU_DEFS, {__index:(key)=>
|
setmetatable(NOMSU_DEFS, {__index:(key)=>
|
||||||
make_node = (start, value, stop)->
|
make_node = (start, value, stop)->
|
||||||
if type(value) == 'table' then error("Not a tuple: #{repr value}")-- = Tuple(value)
|
if type(value) == 'table' then error("Not a tuple: #{repr value}")-- = Tuple(value)
|
||||||
loc = Location(lpeg.userdata.filename, lpeg.userdata.source_code, start, stop)
|
loc = Location(lpeg.userdata.filename, start, stop)
|
||||||
node = Types[key](value, loc)
|
node = Types[key](value, loc)
|
||||||
return node
|
return node
|
||||||
self[key] = make_node
|
self[key] = make_node
|
||||||
@ -247,8 +248,8 @@ class NomsuCompiler
|
|||||||
unless fn_info.isvararg
|
unless fn_info.isvararg
|
||||||
arg_positions = [fn_arg_positions[a] for a in *args]
|
arg_positions = [fn_arg_positions[a] for a in *args]
|
||||||
-- TODO: better error checking?
|
-- TODO: better error checking?
|
||||||
if #arg_positions != #args
|
--if #arg_positions != #args
|
||||||
error("Mismatch in args between lua function's #{repr fn_arg_positions} and stub's #{repr args} for #{repr stub}", 0)
|
-- error("Mismatch in args between lua function's #{repr fn_arg_positions} and stub's #{repr args} for #{repr stub}", 0)
|
||||||
arg_orders[stub] = arg_positions
|
arg_orders[stub] = arg_positions
|
||||||
|
|
||||||
@action_metadata[fn] = {
|
@action_metadata[fn] = {
|
||||||
@ -304,7 +305,8 @@ class NomsuCompiler
|
|||||||
tree = @parse(nomsu_code, filename)
|
tree = @parse(nomsu_code, filename)
|
||||||
assert tree, "Failed to parse: #{nomsu_code}"
|
assert tree, "Failed to parse: #{nomsu_code}"
|
||||||
assert tree.type == "File", "Attempt to run non-file: #{tree.type}"
|
assert tree.type == "File", "Attempt to run non-file: #{tree.type}"
|
||||||
lua = @tree_to_lua(tree)\as_statements!\with_locals_declared!
|
lua = @tree_to_lua(tree)\as_statements!
|
||||||
|
lua\declare_locals!
|
||||||
lua\prepend "-- File: #{filename}\n"
|
lua\prepend "-- File: #{filename}\n"
|
||||||
return @run_lua(lua, filename..".lua")
|
return @run_lua(lua, filename..".lua")
|
||||||
|
|
||||||
@ -353,7 +355,8 @@ class NomsuCompiler
|
|||||||
_chunk_counter += 1
|
_chunk_counter += 1
|
||||||
"<lua chunk ##{_chunk_counter}>"
|
"<lua chunk ##{_chunk_counter}>"
|
||||||
else
|
else
|
||||||
lua.source.text_name..".lua"
|
lua.source.filename..".lua"
|
||||||
|
FILE_CACHE[filename] = tostring(lua)
|
||||||
if type(lua) != 'string'
|
if type(lua) != 'string'
|
||||||
lua, metadata = lua\make_offset_table(filename)
|
lua, metadata = lua\make_offset_table(filename)
|
||||||
LUA_METADATA[lua] = metadata
|
LUA_METADATA[lua] = metadata
|
||||||
@ -373,7 +376,7 @@ class NomsuCompiler
|
|||||||
if tree.type == 'Text' and #tree.value == 1 and type(tree.value[1]) == 'string'
|
if tree.type == 'Text' and #tree.value == 1 and type(tree.value[1]) == 'string'
|
||||||
return tree.value[1]
|
return tree.value[1]
|
||||||
lua = Lua(tree.source, "return ",@tree_to_lua(tree),";")
|
lua = Lua(tree.source, "return ",@tree_to_lua(tree),";")
|
||||||
return @run_lua(lua, tree.source.text_name)
|
return @run_lua(lua, tree.source.filename)
|
||||||
|
|
||||||
tree_to_nomsu: (tree, indentation="", max_line=80, expr_type=nil)=>
|
tree_to_nomsu: (tree, indentation="", max_line=80, expr_type=nil)=>
|
||||||
-- Convert a tree into nomsu code that satisfies the max line requirement or nil
|
-- Convert a tree into nomsu code that satisfies the max line requirement or nil
|
||||||
@ -659,11 +662,11 @@ class NomsuCompiler
|
|||||||
return file_lua
|
return file_lua
|
||||||
|
|
||||||
when "Comment"
|
when "Comment"
|
||||||
return Lua(tree.source, "--"..tree.value\gsub("\n","\n--"))
|
return Lua(tree.source, "--"..tree.value\gsub("\n","\n--").."\n")
|
||||||
|
|
||||||
when "Nomsu"
|
when "Nomsu"
|
||||||
--return Lua(tree.source, repr(tree.value))
|
--return Lua(tree.source, repr(tree.value))
|
||||||
return Lua(tree.source, "nomsu:parse(",tree.source\get_text!,", ",repr(tree.source.text_name),")")
|
return Lua(tree.source, "nomsu:parse(",tree.source\get_text!,", ",repr(tree.source.filename),")")
|
||||||
|
|
||||||
when "Block"
|
when "Block"
|
||||||
block_lua = Lua(tree.source)
|
block_lua = Lua(tree.source)
|
||||||
@ -688,9 +691,9 @@ class NomsuCompiler
|
|||||||
args = [arg for arg in *tree.value when arg.type != "Word"]
|
args = [arg for arg in *tree.value when arg.type != "Word"]
|
||||||
-- Force all compile-time actions to take a tree location
|
-- Force all compile-time actions to take a tree location
|
||||||
if metadata and metadata.arg_orders
|
if metadata and metadata.arg_orders
|
||||||
new_args = [args[p] for p in *metadata.arg_orders[stub]]
|
new_args = [args[p-1] for p in *metadata.arg_orders[stub]]
|
||||||
args = new_args
|
args = new_args
|
||||||
lua = action(unpack(args))
|
lua = action(tree, unpack(args))
|
||||||
remove @compilestack
|
remove @compilestack
|
||||||
return lua
|
return lua
|
||||||
elseif not metadata and @@math_patt\match(stub)
|
elseif not metadata and @@math_patt\match(stub)
|
||||||
@ -1005,15 +1008,51 @@ class NomsuCompiler
|
|||||||
-- Sets up some core functionality
|
-- Sets up some core functionality
|
||||||
get_line_no = -> "nomsu.moon:#{debug_getinfo(2).currentline}"
|
get_line_no = -> "nomsu.moon:#{debug_getinfo(2).currentline}"
|
||||||
nomsu = self
|
nomsu = self
|
||||||
@define_compile_action "immediately %block", get_line_no!, (_block)->
|
@define_compile_action "immediately %block", get_line_no!, (_block)=>
|
||||||
lua = nomsu\tree_to_lua(_block)\as_statements!
|
lua = nomsu\tree_to_lua(_block)\as_statements!
|
||||||
lua\declare_locals!
|
lua\declare_locals!
|
||||||
nomsu\run_lua(lua)
|
nomsu\run_lua(lua)
|
||||||
return Lua(_block.source, "if IMMEDIATE then\n", lua, "\nend")
|
return Lua(@source, "if IMMEDIATE then\n", lua, "\nend")
|
||||||
|
|
||||||
@define_compile_action "lua> %code", get_line_no!, (_code)->
|
@define_compile_action "Lua %code", get_line_no!, (_code)=>
|
||||||
if _code.type != "Text"
|
if _code.type != "Text"
|
||||||
return LuaValue(_code.source, "nomsu:run_lua(",nomsu\tree_to_lua(_code),")")
|
return LuaValue(@source, "Lua(", repr(_code.source),", ",nomsu\tree_to_lua(_code),")")
|
||||||
|
|
||||||
|
lua = LuaValue(@source, "Lua(", repr(_code.source))
|
||||||
|
for bit in *_code.value
|
||||||
|
lua\append ", "
|
||||||
|
if type(bit) == "string"
|
||||||
|
lua\append repr(bit)
|
||||||
|
else
|
||||||
|
bit_lua = nomsu\tree_to_lua bit
|
||||||
|
unless bit_lua.is_value
|
||||||
|
line, src = bit.source\get_line!, bit.source\get_text!
|
||||||
|
error "#{line}: Cannot use #{colored.yellow src} as a string interpolation value, since it's not an expression.", 0
|
||||||
|
lua\append bit_lua
|
||||||
|
lua\append ")"
|
||||||
|
return lua
|
||||||
|
|
||||||
|
@define_compile_action "LuaValue %code", get_line_no!, (_code)=>
|
||||||
|
if _code.type != "Text"
|
||||||
|
return LuaValue(@source, "LuaValue(", repr(_code.source),", ",nomsu\tree_to_lua(_code),")")
|
||||||
|
|
||||||
|
lua = LuaValue(@source, "LuaValue(", repr(_code.source))
|
||||||
|
for bit in *_code.value
|
||||||
|
lua\append ", "
|
||||||
|
if type(bit) == "string"
|
||||||
|
lua\append repr(bit)
|
||||||
|
else
|
||||||
|
bit_lua = nomsu\tree_to_lua bit
|
||||||
|
unless bit_lua.is_value
|
||||||
|
line, src = bit.source\get_line!, bit.source\get_text!
|
||||||
|
error "#{line}: Cannot use #{colored.yellow src} as a string interpolation value, since it's not an expression.", 0
|
||||||
|
lua\append bit_lua
|
||||||
|
lua\append ")"
|
||||||
|
return lua
|
||||||
|
|
||||||
|
@define_compile_action "lua> %code", get_line_no!, (_code)=>
|
||||||
|
if _code.type != "Text"
|
||||||
|
return LuaValue(@source, "nomsu:run_lua(",nomsu\tree_to_lua(_code),")")
|
||||||
|
|
||||||
lua = Lua(_code.source)
|
lua = Lua(_code.source)
|
||||||
for bit in *_code.value
|
for bit in *_code.value
|
||||||
@ -1027,11 +1066,11 @@ class NomsuCompiler
|
|||||||
lua\append bit_lua
|
lua\append bit_lua
|
||||||
return lua
|
return lua
|
||||||
|
|
||||||
@define_compile_action "=lua %code", get_line_no!, (_code)->
|
@define_compile_action "=lua %code", get_line_no!, (_code)=>
|
||||||
if _code.type != "Text"
|
if _code.type != "Text"
|
||||||
return LuaValue(_code.source, "nomsu:run_lua(",nomsu\tree_to_lua(_code),")")
|
return LuaValue(@source, "nomsu:run_lua(",nomsu\tree_to_lua(_code),")")
|
||||||
|
|
||||||
lua = LuaValue(_code.source)
|
lua = LuaValue(@source)
|
||||||
for bit in *_code.value
|
for bit in *_code.value
|
||||||
if type(bit) == "string"
|
if type(bit) == "string"
|
||||||
lua\append bit
|
lua\append bit
|
||||||
@ -1043,18 +1082,16 @@ class NomsuCompiler
|
|||||||
lua\append bit_lua
|
lua\append bit_lua
|
||||||
return lua
|
return lua
|
||||||
|
|
||||||
@define_compile_action "!! code location !!", get_line_no!, ->
|
@define_compile_action "!! code location !!", get_line_no!, =>
|
||||||
tree = nomsu.compilestack[#nomsu.compilestack-1]
|
return LuaValue(@source, repr(@source))
|
||||||
return LuaValue(tree.source, repr(tostring(tree.source)))
|
|
||||||
|
|
||||||
@define_action "run file %filename", get_line_no!, (_filename)->
|
@define_action "run file %filename", get_line_no!, (_filename)->
|
||||||
return nomsu\run_file(_filename)
|
return nomsu\run_file(_filename)
|
||||||
|
|
||||||
@define_compile_action "use %filename", get_line_no!, (_filename)->
|
@define_compile_action "use %filename", get_line_no!, (_filename)=>
|
||||||
tree = nomsu.compilestack[#nomsu.compilestack-1]
|
|
||||||
filename = nomsu\tree_to_value(_filename)
|
filename = nomsu\tree_to_value(_filename)
|
||||||
nomsu\use_file(filename)
|
nomsu\use_file(filename)
|
||||||
return LuaValue(tree.source, "nomsu:use_file(#{repr filename})")
|
return LuaValue(@source, "nomsu:use_file(#{repr filename})")
|
||||||
|
|
||||||
-- 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
|
||||||
|
Loading…
Reference in New Issue
Block a user