Fixing up error reporting and ripping out LDT-specific code (now a
debugger can be provided by a command line flag)
This commit is contained in:
parent
de34592dbe
commit
fa72d7eeb1
@ -173,11 +173,11 @@ do
|
|||||||
__init = function(self, source, ...)
|
__init = function(self, source, ...)
|
||||||
self.source = source
|
self.source = source
|
||||||
self.bits, self.indents, self.current_indent = { }, { }, 0
|
self.bits, self.indents, self.current_indent = { }, { }, 0
|
||||||
self:append(...)
|
|
||||||
if type(self.source) == 'string' then
|
if type(self.source) == 'string' then
|
||||||
self.source = Source:from_string(self.source)
|
self.source = Source:from_string(self.source)
|
||||||
end
|
end
|
||||||
return assert(self.source and Source:is_instance(self.source))
|
assert(self.source and Source:is_instance(self.source))
|
||||||
|
return self:append(...)
|
||||||
end,
|
end,
|
||||||
__base = _base_0,
|
__base = _base_0,
|
||||||
__name = "Code"
|
__name = "Code"
|
||||||
|
@ -46,10 +46,10 @@ class Source
|
|||||||
class Code
|
class Code
|
||||||
new: (@source, ...)=>
|
new: (@source, ...)=>
|
||||||
@bits, @indents, @current_indent = {}, {}, 0
|
@bits, @indents, @current_indent = {}, {}, 0
|
||||||
@append(...)
|
|
||||||
if type(@source) == 'string'
|
if type(@source) == 'string'
|
||||||
@source = Source\from_string(@source)
|
@source = Source\from_string(@source)
|
||||||
assert(@source and Source\is_instance(@source))
|
assert(@source and Source\is_instance(@source))
|
||||||
|
@append(...)
|
||||||
|
|
||||||
append: (...)=>
|
append: (...)=>
|
||||||
n = select("#",...)
|
n = select("#",...)
|
||||||
|
@ -9,6 +9,7 @@ use "core/operators.nom"
|
|||||||
# List/dict functions:
|
# List/dict functions:
|
||||||
|
|
||||||
# Indexing
|
# Indexing
|
||||||
|
test: assume: (2nd to last in [1,2,3,4,5]) is 4
|
||||||
compile [..]
|
compile [..]
|
||||||
%index st to last in %list, %index nd to last in %list, %index rd to last in %list
|
%index st to last in %list, %index nd to last in %list, %index rd to last in %list
|
||||||
%index th to last in %list
|
%index th to last in %list
|
||||||
@ -20,11 +21,13 @@ parse [last in %list] as: 1st to last in %list
|
|||||||
parse [first in %list] as: %list.1
|
parse [first in %list] as: %list.1
|
||||||
|
|
||||||
# Membership testing
|
# Membership testing
|
||||||
|
test: assume: 3 is in [1,2,3,4,5]
|
||||||
action [%item is in %list, %list contains %item, %list has %item]
|
action [%item is in %list, %list contains %item, %list has %item]
|
||||||
for %key = %value in %list
|
for %key = %value in %list
|
||||||
if (%key is %item): return (yes)
|
if (%key is %item): return (yes)
|
||||||
return (no)
|
return (no)
|
||||||
|
|
||||||
|
test: assume: 99 isn't in [1,2,3]
|
||||||
action [..]
|
action [..]
|
||||||
%item isn't in %list, %item is not in %list
|
%item isn't in %list, %item is not in %list
|
||||||
%list doesn't contain %item, %list does not contain %item
|
%list doesn't contain %item, %list does not contain %item
|
||||||
@ -34,9 +37,13 @@ action [..]
|
|||||||
if (%key is %item): return (no)
|
if (%key is %item): return (no)
|
||||||
return (yes)
|
return (yes)
|
||||||
|
|
||||||
|
test: assume: {x:no} has key "x"
|
||||||
parse [%list has key %index, %list has index %index] as
|
parse [%list has key %index, %list has index %index] as
|
||||||
%list.%index != (nil)
|
%list.%index != (nil)
|
||||||
|
|
||||||
|
test
|
||||||
|
assume: {x:no} doesn't have key "y"
|
||||||
|
assume: not: {x:no} doesn't have key "x"
|
||||||
parse [..]
|
parse [..]
|
||||||
%list doesn't have key %index, %list does not have key %index
|
%list doesn't have key %index, %list does not have key %index
|
||||||
%list doesn't have index %index, %list does not have index %index
|
%list doesn't have index %index, %list does not have index %index
|
||||||
@ -46,6 +53,15 @@ parse [..]
|
|||||||
compile [number of keys in %list] to
|
compile [number of keys in %list] to
|
||||||
Lua value "utils.size(\(%list as lua expr))"
|
Lua value "utils.size(\(%list as lua expr))"
|
||||||
|
|
||||||
|
test
|
||||||
|
%list <- [1,2,3,4,5]
|
||||||
|
append 6 to %list
|
||||||
|
assume: (last in %list) is 6
|
||||||
|
pop from %list
|
||||||
|
assume: (last in %list) is 5
|
||||||
|
remove index 1 from %list
|
||||||
|
assume: (first in %list) is 2
|
||||||
|
|
||||||
compile [append %item to %list, add %item to %list, to %list add %item, to %list append %item] to
|
compile [append %item to %list, add %item to %list, to %list add %item, to %list append %item] to
|
||||||
Lua "table.insert(\(%list as lua expr), \(%item as lua expr))"
|
Lua "table.insert(\(%list as lua expr), \(%item as lua expr))"
|
||||||
|
|
||||||
@ -58,6 +74,7 @@ compile [remove index %index from %list] to
|
|||||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
|
||||||
# List Comprehension
|
# List Comprehension
|
||||||
|
test: assume: ((% * %) for % in [1,2,3]) = [1,4,9]
|
||||||
parse [%expression for %item in %iterable] as
|
parse [%expression for %item in %iterable] as
|
||||||
result of
|
result of
|
||||||
%comprehension <- []
|
%comprehension <- []
|
||||||
@ -76,9 +93,11 @@ parse [..]
|
|||||||
return %comprehension
|
return %comprehension
|
||||||
|
|
||||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
test: assume: ((% * %) for % in 1 to 3) = [1,4,9]
|
||||||
parse [%expression for %var in %start to %stop] as
|
parse [%expression for %var in %start to %stop] as
|
||||||
%expression for %var in %start to %stop via 1
|
%expression for %var in %start to %stop via 1
|
||||||
|
|
||||||
|
test: assume: ("\%k,\%v" for %k = %v in {x:1}) = ["x,1"]
|
||||||
parse [..]
|
parse [..]
|
||||||
%expression for %key = %value in %iterable
|
%expression for %key = %value in %iterable
|
||||||
%expression for (%key,%value) in %iterable
|
%expression for (%key,%value) in %iterable
|
||||||
@ -90,6 +109,7 @@ parse [..]
|
|||||||
return %comprehension
|
return %comprehension
|
||||||
|
|
||||||
# Dict comprehensions
|
# Dict comprehensions
|
||||||
|
test: assume: ((% * %) = % for % in [1,2,3]) = {1:1,4:2,9:3}
|
||||||
parse [..]
|
parse [..]
|
||||||
%key = %value for %item in %iterable
|
%key = %value for %item in %iterable
|
||||||
(%key,%value) for %item in %iterable
|
(%key,%value) for %item in %iterable
|
||||||
@ -100,6 +120,7 @@ parse [..]
|
|||||||
%comprehension.%key <- %value
|
%comprehension.%key <- %value
|
||||||
return %comprehension
|
return %comprehension
|
||||||
|
|
||||||
|
test: assume: (%k = (%v * %v) for %k = %v in {x:1,y:2,z:3}) = {x:1,y:4,z:9}
|
||||||
parse [..]
|
parse [..]
|
||||||
%key = %value for %src_key = %src_value in %iterable
|
%key = %value for %src_key = %src_value in %iterable
|
||||||
(%key,%value) for (%src_key,%src_value) in %iterable
|
(%key,%value) for (%src_key,%src_value) in %iterable
|
||||||
@ -110,7 +131,23 @@ parse [..]
|
|||||||
%comprehension.%key <- %value
|
%comprehension.%key <- %value
|
||||||
return %comprehension
|
return %comprehension
|
||||||
|
|
||||||
|
parse [..]
|
||||||
|
%key = %value for %item in %start to %stop via %step
|
||||||
|
(%key,%value) for %item in %start to %stop via %step
|
||||||
|
..as
|
||||||
|
result of
|
||||||
|
%comprehension <- {}
|
||||||
|
for %item in %start to %stop via %step
|
||||||
|
%comprehension.%key <- %value
|
||||||
|
return %comprehension
|
||||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
test: assume: ((% * %) = % for % in 1 to 3) = {1:1,4:2,9:3}
|
||||||
|
parse [..]
|
||||||
|
%key = %value for %item in %start to %stop
|
||||||
|
(%key,%value) for %item in %start to %stop
|
||||||
|
..as: %key = %value for %item in %start to %stop via 1
|
||||||
|
|
||||||
|
test: assume: ([[1,2],[3,4]] flattened) = [1,2,3,4]
|
||||||
action [%lists flattened]
|
action [%lists flattened]
|
||||||
%flat <- []
|
%flat <- []
|
||||||
for %list in %lists
|
for %list in %lists
|
||||||
@ -118,14 +155,24 @@ action [%lists flattened]
|
|||||||
add %item to %flat
|
add %item to %flat
|
||||||
return %flat
|
return %flat
|
||||||
|
|
||||||
|
test: assume: (entries in {x:1}) = [{key:"x",value:1}]
|
||||||
parse [entries in %dict] as: {key:%k, value:%v} for %k = %v in %dict
|
parse [entries in %dict] as: {key:%k, value:%v} for %k = %v in %dict
|
||||||
|
|
||||||
|
test: assume: (keys in {x:1}) = ["x"]
|
||||||
parse [keys in %dict, keys of %dict] as: %k for %k = %v in %dict
|
parse [keys in %dict, keys of %dict] as: %k for %k = %v in %dict
|
||||||
|
|
||||||
|
test: assume: (values in {x:1}) = [1]
|
||||||
parse [values in %dict, values of %dict] as: %v for %k = %v in %dict
|
parse [values in %dict, values of %dict] as: %v for %k = %v in %dict
|
||||||
|
|
||||||
# Metatable stuff
|
# Metatable stuff
|
||||||
|
test
|
||||||
|
%t <- {}
|
||||||
|
set %t's metatable to {__tostring:[%]->"XXX"}
|
||||||
|
assume: "\%t" = "XXX"
|
||||||
compile [set %dict's metatable to %metatable] to
|
compile [set %dict's metatable to %metatable] to
|
||||||
Lua "setmetatable(\(%dict as lua expr), \(%metatable as lua expr));"
|
Lua "setmetatable(\(%dict as lua expr), \(%metatable as lua expr));"
|
||||||
|
|
||||||
|
test: assume: ({} with fallback % -> (% + 1)).10 = 11
|
||||||
compile [%dict with fallback %key -> %value] to
|
compile [%dict with fallback %key -> %value] to
|
||||||
Lua value ".."
|
Lua value ".."
|
||||||
setmetatable(\(%dict as lua expr), {__index=function(self, \(%key as lua expr))
|
setmetatable(\(%dict as lua expr), {__index=function(self, \(%key as lua expr))
|
||||||
@ -134,6 +181,8 @@ compile [%dict with fallback %key -> %value] to
|
|||||||
return value
|
return value
|
||||||
end})
|
end})
|
||||||
|
|
||||||
|
test: assume: 1 is 2
|
||||||
|
|
||||||
# Sorting
|
# Sorting
|
||||||
compile [sort %items] to: Lua "table.sort(\(%items as lua expr));"
|
compile [sort %items] to: Lua "table.sort(\(%items as lua expr));"
|
||||||
parse [..]
|
parse [..]
|
||||||
|
@ -8,17 +8,17 @@ compile [traceback %] to: Lua value "debug.traceback('', \(% as lua expr))"
|
|||||||
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(nomsu:tree_to_nomsu(\%condition));"
|
lua> "local \%assumption = 'Assumption failed: '..tostring(nomsu:tree_to_nomsu(\%condition))"
|
||||||
return
|
return
|
||||||
Lua ".."
|
Lua ".."
|
||||||
if not \(%condition as lua expr) then
|
if not \(%condition as lua expr) then
|
||||||
error(\(quote "\%assumption"), 0);
|
error(\(quote "\%assumption"), 0)
|
||||||
end
|
end
|
||||||
|
|
||||||
compile [assume %condition or barf %message] to
|
compile [assume %condition or barf %message] to
|
||||||
Lua ".."
|
Lua ".."
|
||||||
if not \(%condition as lua expr) then
|
if not \(%condition as lua expr) then
|
||||||
error(\(%message as lua expr), 0);
|
error(\(%message as lua expr), 0)
|
||||||
end
|
end
|
||||||
|
|
||||||
# Try/except
|
# Try/except
|
||||||
|
@ -124,6 +124,11 @@ compile [parse %actions as %body] to
|
|||||||
return ret
|
return ret
|
||||||
|
|
||||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
lua> ".."
|
||||||
|
COMPILE_ACTIONS["% as lua expr"] = function(nomsu, tree, _t)
|
||||||
|
return LuaCode.Value(tree.source, "nomsu:compile(", nomsu:compile(_t):as_expr(), "):as_expr()")
|
||||||
|
end
|
||||||
|
#
|
||||||
compile [%tree as lua expr] to
|
compile [%tree as lua expr] to
|
||||||
Lua value "nomsu:compile(\(=lua "nomsu:compile(\%tree):as_expr()")):as_expr()"
|
Lua value "nomsu:compile(\(=lua "nomsu:compile(\%tree):as_expr()")):as_expr()"
|
||||||
|
|
||||||
|
@ -163,6 +163,7 @@ compile [%x ARSHIFT %shift, %x >> %shift] to: Lua value "(\(%x as lua expr) >> \
|
|||||||
# Unary operators
|
# Unary operators
|
||||||
compile [- %] to: Lua value "(- \(% as lua expr))"
|
compile [- %] to: Lua value "(- \(% as lua expr))"
|
||||||
compile [not %] to: Lua value "(not \(% as lua expr))"
|
compile [not %] to: Lua value "(not \(% as lua expr))"
|
||||||
|
test: assume: (length of [1,2,3]) = 3
|
||||||
compile [length of %list, || %list ||] to: Lua value "(#\(%list as lua expr))"
|
compile [length of %list, || %list ||] to: Lua value "(#\(%list as lua expr))"
|
||||||
|
|
||||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
@ -45,25 +45,31 @@ debug.getinfo = function(thread, f, what)
|
|||||||
info.lastlinedefined = assert(map[info.lastlinedefined])
|
info.lastlinedefined = assert(map[info.lastlinedefined])
|
||||||
end
|
end
|
||||||
info.short_src = info.source:match('@([^[]*)') or info.short_src
|
info.short_src = info.source:match('@([^[]*)') or info.short_src
|
||||||
|
if info.name then
|
||||||
|
do
|
||||||
|
local tmp = info.name:match("^A_([a-zA-Z0-9_]*)$")
|
||||||
|
if tmp then
|
||||||
|
info.name = tmp:gsub("_", " "):gsub("x([0-9A-F][0-9A-F])", function(self)
|
||||||
|
return string.char(tonumber(self, 16))
|
||||||
|
end)
|
||||||
|
else
|
||||||
|
info.name = info.name
|
||||||
|
end
|
||||||
|
end
|
||||||
|
else
|
||||||
|
info.name = "main chunk"
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
return info
|
return info
|
||||||
end
|
end
|
||||||
local print_error
|
local print_error
|
||||||
print_error = function(error_message, stack_offset)
|
print_error = function(error_message, start_fn, stop_fn)
|
||||||
if stack_offset == nil then
|
|
||||||
stack_offset = 3
|
|
||||||
end
|
|
||||||
io.stderr:write(tostring(colored.red("ERROR:")) .. " " .. tostring(colored.bright(colored.red((error_message or "")))) .. "\n")
|
io.stderr:write(tostring(colored.red("ERROR:")) .. " " .. tostring(colored.bright(colored.red((error_message or "")))) .. "\n")
|
||||||
io.stderr:write("stack traceback:\n")
|
io.stderr:write("stack traceback:\n")
|
||||||
local get_line
|
local level = 1
|
||||||
get_line = function(file, line_no)
|
local found_start = false
|
||||||
local start = LINE_STARTS[file][line_no] or 1
|
|
||||||
local stop = (LINE_STARTS[file][line_no + 1] or 0) - 1
|
|
||||||
return file:sub(start, stop)
|
|
||||||
end
|
|
||||||
local level = stack_offset
|
|
||||||
while true do
|
while true do
|
||||||
local _continue_0 = false
|
local _continue_0 = false
|
||||||
repeat
|
repeat
|
||||||
@ -71,10 +77,14 @@ print_error = function(error_message, stack_offset)
|
|||||||
if not calling_fn then
|
if not calling_fn then
|
||||||
break
|
break
|
||||||
end
|
end
|
||||||
if calling_fn.func == run then
|
level = level + 1
|
||||||
|
if not (found_start) then
|
||||||
|
if calling_fn.func == start_fn then
|
||||||
|
found_start = true
|
||||||
|
end
|
||||||
|
_continue_0 = true
|
||||||
break
|
break
|
||||||
end
|
end
|
||||||
level = level + 1
|
|
||||||
local name = calling_fn.name and "function '" .. tostring(calling_fn.name) .. "'" or nil
|
local name = calling_fn.name and "function '" .. tostring(calling_fn.name) .. "'" or nil
|
||||||
if calling_fn.linedefined == 0 then
|
if calling_fn.linedefined == 0 then
|
||||||
name = "main chunk"
|
name = "main chunk"
|
||||||
@ -98,8 +108,8 @@ print_error = function(error_message, stack_offset)
|
|||||||
end
|
end
|
||||||
local filename, start, stop = calling_fn.source:match('@([^[]*)%[([0-9]+):([0-9]+)]')
|
local filename, start, stop = calling_fn.source:match('@([^[]*)%[([0-9]+):([0-9]+)]')
|
||||||
assert(filename)
|
assert(filename)
|
||||||
local file = read_file(filename):sub(tonumber(start), tonumber(stop))
|
local file = files.read(filename)
|
||||||
local err_line = get_line(file, calling_fn.currentline):sub(1, -2)
|
local err_line = files.get_line(file, calling_fn.currentline)
|
||||||
local offending_statement = colored.bright(colored.red(err_line:match("^[ ]*(.*)")))
|
local offending_statement = colored.bright(colored.red(err_line:match("^[ ]*(.*)")))
|
||||||
if calling_fn.name then
|
if calling_fn.name then
|
||||||
do
|
do
|
||||||
@ -119,7 +129,7 @@ print_error = function(error_message, stack_offset)
|
|||||||
else
|
else
|
||||||
local file
|
local file
|
||||||
ok, file = pcall(function()
|
ok, file = pcall(function()
|
||||||
return read_file(calling_fn.short_src)
|
return files.read(calling_fn.short_src)
|
||||||
end)
|
end)
|
||||||
if not ok then
|
if not ok then
|
||||||
file = nil
|
file = nil
|
||||||
@ -128,11 +138,12 @@ print_error = function(error_message, stack_offset)
|
|||||||
if name == nil then
|
if name == nil then
|
||||||
local search_level = level
|
local search_level = level
|
||||||
local _info = debug.getinfo(search_level)
|
local _info = debug.getinfo(search_level)
|
||||||
while _info and (_info.func == pcall or _info.func == xpcall) do
|
while true do
|
||||||
search_level = search_level + 1
|
search_level = search_level + 1
|
||||||
_info = debug.getinfo(search_level)
|
_info = debug.getinfo(search_level)
|
||||||
|
if not (_info) then
|
||||||
|
break
|
||||||
end
|
end
|
||||||
if _info then
|
|
||||||
for i = 1, 999 do
|
for i = 1, 999 do
|
||||||
local varname, val = debug.getlocal(search_level, i)
|
local varname, val = debug.getlocal(search_level, i)
|
||||||
if not varname then
|
if not varname then
|
||||||
@ -177,16 +188,19 @@ print_error = function(error_message, stack_offset)
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
if file then
|
if file then
|
||||||
local err_line = get_line(file, line_num):sub(1, -2)
|
local err_line = files.get_line(file, line_num)
|
||||||
local offending_statement = colored.bright(colored.red(err_line:match("^[ ]*(.*)$")))
|
local offending_statement = colored.bright(colored.red(err_line:match("^[ ]*(.*)$")))
|
||||||
line = line .. ("\n " .. offending_statement)
|
line = line .. ("\n " .. offending_statement)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
io.stderr:write(" " .. tostring(line) .. "\n")
|
io.stderr:write(line, "\n")
|
||||||
if calling_fn.istailcall then
|
if calling_fn.istailcall then
|
||||||
io.stderr:write(" " .. tostring(colored.dim(colored.white(" (...tail calls...)"))) .. "\n")
|
io.stderr:write(" " .. tostring(colored.dim(colored.white(" (...tail calls...)"))) .. "\n")
|
||||||
end
|
end
|
||||||
|
if calling_fn.func == stop_fn then
|
||||||
|
break
|
||||||
|
end
|
||||||
_continue_0 = true
|
_continue_0 = true
|
||||||
until true
|
until true
|
||||||
if not _continue_0 then
|
if not _continue_0 then
|
||||||
@ -195,18 +209,17 @@ print_error = function(error_message, stack_offset)
|
|||||||
end
|
end
|
||||||
return io.stderr:flush()
|
return io.stderr:flush()
|
||||||
end
|
end
|
||||||
|
local guard
|
||||||
|
guard = function(fn)
|
||||||
local error_handler
|
local error_handler
|
||||||
error_handler = function(error_message)
|
error_handler = function(error_message)
|
||||||
print_error(error_message)
|
print_error(error_message, error_handler, fn)
|
||||||
local EXIT_FAILURE = 1
|
local EXIT_FAILURE = 1
|
||||||
return os.exit(EXIT_FAILURE)
|
return os.exit(EXIT_FAILURE)
|
||||||
end
|
end
|
||||||
local run_safely
|
|
||||||
run_safely = function(fn)
|
|
||||||
return xpcall(fn, error_handler)
|
return xpcall(fn, error_handler)
|
||||||
end
|
end
|
||||||
return {
|
return {
|
||||||
run_safely = run_safely,
|
guard = guard,
|
||||||
print_error = print_error,
|
print_error = print_error
|
||||||
error_handler = error_handler
|
|
||||||
}
|
}
|
||||||
|
@ -31,24 +31,28 @@ debug.getinfo = (thread,f,what)->
|
|||||||
if info.lastlinedefined
|
if info.lastlinedefined
|
||||||
info.lastlinedefined = assert(map[info.lastlinedefined])
|
info.lastlinedefined = assert(map[info.lastlinedefined])
|
||||||
info.short_src = info.source\match('@([^[]*)') or info.short_src
|
info.short_src = info.source\match('@([^[]*)') or info.short_src
|
||||||
|
-- TODO: get name properly
|
||||||
|
info.name = if info.name
|
||||||
|
if tmp = info.name\match("^A_([a-zA-Z0-9_]*)$")
|
||||||
|
tmp\gsub("_"," ")\gsub("x([0-9A-F][0-9A-F])", => string.char(tonumber(@, 16)))
|
||||||
|
else info.name
|
||||||
|
else "main chunk"
|
||||||
return info
|
return info
|
||||||
|
|
||||||
print_error = (error_message, stack_offset=3)->
|
print_error = (error_message, start_fn, stop_fn)->
|
||||||
io.stderr\write("#{colored.red "ERROR:"} #{colored.bright colored.red (error_message or "")}\n")
|
io.stderr\write("#{colored.red "ERROR:"} #{colored.bright colored.red (error_message or "")}\n")
|
||||||
io.stderr\write("stack traceback:\n")
|
io.stderr\write("stack traceback:\n")
|
||||||
|
|
||||||
get_line = (file, line_no)->
|
level = 1
|
||||||
start = LINE_STARTS[file][line_no] or 1
|
found_start = false
|
||||||
stop = (LINE_STARTS[file][line_no+1] or 0) - 1
|
|
||||||
return file\sub(start, stop)
|
|
||||||
|
|
||||||
level = stack_offset
|
|
||||||
while true
|
while true
|
||||||
-- TODO: reduce duplicate code
|
-- TODO: reduce duplicate code
|
||||||
calling_fn = debug_getinfo(level)
|
calling_fn = debug_getinfo(level)
|
||||||
if not calling_fn then break
|
if not calling_fn then break
|
||||||
if calling_fn.func == run then break
|
|
||||||
level += 1
|
level += 1
|
||||||
|
unless found_start
|
||||||
|
if calling_fn.func == start_fn then found_start = true
|
||||||
|
continue
|
||||||
name = calling_fn.name and "function '#{calling_fn.name}'" or nil
|
name = calling_fn.name and "function '#{calling_fn.name}'" or nil
|
||||||
if calling_fn.linedefined == 0 then name = "main chunk"
|
if calling_fn.linedefined == 0 then name = "main chunk"
|
||||||
if name == "run_lua_fn" then continue
|
if name == "run_lua_fn" then continue
|
||||||
@ -63,8 +67,8 @@ print_error = (error_message, stack_offset=3)->
|
|||||||
--calling_fn.short_src = calling_fn.source\match('"([^[]*)')
|
--calling_fn.short_src = calling_fn.source\match('"([^[]*)')
|
||||||
filename,start,stop = calling_fn.source\match('@([^[]*)%[([0-9]+):([0-9]+)]')
|
filename,start,stop = calling_fn.source\match('@([^[]*)%[([0-9]+):([0-9]+)]')
|
||||||
assert(filename)
|
assert(filename)
|
||||||
file = read_file(filename)\sub(tonumber(start),tonumber(stop))
|
file = files.read(filename)
|
||||||
err_line = get_line(file, calling_fn.currentline)\sub(1,-2)
|
err_line = files.get_line(file, calling_fn.currentline)
|
||||||
offending_statement = colored.bright(colored.red(err_line\match("^[ ]*(.*)")))
|
offending_statement = colored.bright(colored.red(err_line\match("^[ ]*(.*)")))
|
||||||
-- TODO: get name properly
|
-- TODO: get name properly
|
||||||
name = if calling_fn.name
|
name = if calling_fn.name
|
||||||
@ -74,16 +78,16 @@ print_error = (error_message, stack_offset=3)->
|
|||||||
else "main chunk"
|
else "main chunk"
|
||||||
line = colored.yellow("#{filename}:#{calling_fn.currentline} in #{name}\n #{offending_statement}")
|
line = colored.yellow("#{filename}:#{calling_fn.currentline} in #{name}\n #{offending_statement}")
|
||||||
else
|
else
|
||||||
ok, file = pcall ->read_file(calling_fn.short_src)
|
ok, file = pcall ->files.read(calling_fn.short_src)
|
||||||
if not ok then file = nil
|
if not ok then file = nil
|
||||||
local line_num
|
local line_num
|
||||||
if name == nil
|
if name == nil
|
||||||
search_level = level
|
search_level = level
|
||||||
_info = debug.getinfo(search_level)
|
_info = debug.getinfo(search_level)
|
||||||
while _info and (_info.func == pcall or _info.func == xpcall)
|
while true
|
||||||
search_level += 1
|
search_level += 1
|
||||||
_info = debug.getinfo(search_level)
|
_info = debug.getinfo(search_level)
|
||||||
if _info
|
break unless _info
|
||||||
for i=1,999
|
for i=1,999
|
||||||
varname, val = debug.getlocal(search_level, i)
|
varname, val = debug.getlocal(search_level, i)
|
||||||
if not varname then break
|
if not varname then break
|
||||||
@ -113,21 +117,21 @@ print_error = (error_message, stack_offset=3)->
|
|||||||
line = colored.blue("#{calling_fn.short_src}:#{calling_fn.currentline} in #{name or '?'}")
|
line = colored.blue("#{calling_fn.short_src}:#{calling_fn.currentline} in #{name or '?'}")
|
||||||
|
|
||||||
if file
|
if file
|
||||||
err_line = get_line(file, line_num)\sub(1,-2)
|
err_line = files.get_line(file, line_num)
|
||||||
offending_statement = colored.bright(colored.red(err_line\match("^[ ]*(.*)$")))
|
offending_statement = colored.bright(colored.red(err_line\match("^[ ]*(.*)$")))
|
||||||
line ..= "\n "..offending_statement
|
line ..= "\n "..offending_statement
|
||||||
io.stderr\write(" #{line}\n")
|
io.stderr\write(line,"\n")
|
||||||
if calling_fn.istailcall
|
if calling_fn.istailcall
|
||||||
io.stderr\write(" #{colored.dim colored.white " (...tail calls...)"}\n")
|
io.stderr\write(" #{colored.dim colored.white " (...tail calls...)"}\n")
|
||||||
|
if calling_fn.func == stop_fn then break
|
||||||
|
|
||||||
io.stderr\flush!
|
io.stderr\flush!
|
||||||
|
|
||||||
|
guard = (fn)->
|
||||||
error_handler = (error_message)->
|
error_handler = (error_message)->
|
||||||
print_error error_message
|
print_error error_message, error_handler, fn
|
||||||
EXIT_FAILURE = 1
|
EXIT_FAILURE = 1
|
||||||
os.exit(EXIT_FAILURE)
|
os.exit(EXIT_FAILURE)
|
||||||
|
|
||||||
run_safely = (fn)->
|
|
||||||
xpcall(fn, error_handler)
|
xpcall(fn, error_handler)
|
||||||
|
|
||||||
return {:run_safely, :print_error, :error_handler}
|
return {:guard, :print_error}
|
||||||
|
33
nomsu.lua
33
nomsu.lua
@ -53,13 +53,14 @@ end
|
|||||||
local EXIT_SUCCESS, EXIT_FAILURE = 0, 1
|
local EXIT_SUCCESS, EXIT_FAILURE = 0, 1
|
||||||
local usage = [=[Nomsu Compiler
|
local usage = [=[Nomsu Compiler
|
||||||
|
|
||||||
Usage: (nomsu | lua nomsu.lua | moon nomsu.moon) [-V version] [-O] [-v] [-c] [-s] [-I file] [--help | -h] [--version] [file [nomsu args...]]
|
Usage: (nomsu | lua nomsu.lua | moon nomsu.moon) [-V version] [-O] [-v] [-c] [-s] [-t] [-I file] [--help | -h] [--version] [file [nomsu args...]]
|
||||||
|
|
||||||
OPTIONS
|
OPTIONS
|
||||||
-O Run the compiler in optimized mode (use precompiled .lua versions of Nomsu files, when available).
|
-O Run the compiler in optimized mode (use precompiled .lua versions of Nomsu files, when available).
|
||||||
-v Verbose: print compiled lua code.
|
-v Verbose: print compiled lua code.
|
||||||
-c Compile the input files into a .lua files.
|
-c Compile the input files into a .lua files.
|
||||||
-s Check the input files for syntax errors.
|
-s Check the input files for syntax errors.
|
||||||
|
-t Run tests.
|
||||||
-I <file> Add an additional input file or directory.
|
-I <file> Add an additional input file or directory.
|
||||||
-h/--help Print this message.
|
-h/--help Print this message.
|
||||||
--version Print the version number and exit.
|
--version Print the version number and exit.
|
||||||
@ -93,9 +94,11 @@ local parser = re.compile([[ args <- {| (flag ";")* (({~ file ~} -> add_file)
|
|||||||
/ ("-I" (";")? ({~ file ~} -> add_file))
|
/ ("-I" (";")? ({~ file ~} -> add_file))
|
||||||
/ ({:check_syntax: ("-s" -> true):})
|
/ ({:check_syntax: ("-s" -> true):})
|
||||||
/ ({:compile: ("-c" -> true):})
|
/ ({:compile: ("-c" -> true):})
|
||||||
|
/ {:run_tests: ("-t" -> true) :}
|
||||||
/ {:verbose: ("-v" -> true) :}
|
/ {:verbose: ("-v" -> true) :}
|
||||||
/ {:help: (("-h" / "--help") -> true) :}
|
/ {:help: (("-h" / "--help") -> true) :}
|
||||||
/ {:version: ("--version" -> true) :}
|
/ {:version: ("--version" -> true) :}
|
||||||
|
/ {:debugger: ("-d" (";")? {([^;])*}) :}
|
||||||
/ {:requested_version: "-V" ((";")? {([0-9.])+})? :}
|
/ {:requested_version: "-V" ((";")? {([0-9.])+})? :}
|
||||||
file <- ("-" -> "stdin") / {[^;]+}
|
file <- ("-" -> "stdin") / {[^;]+}
|
||||||
]], {
|
]], {
|
||||||
@ -153,6 +156,13 @@ run = function()
|
|||||||
end
|
end
|
||||||
return true
|
return true
|
||||||
end
|
end
|
||||||
|
local tests = { }
|
||||||
|
if args.run_tests then
|
||||||
|
nomsu.COMPILE_ACTIONS["test %"] = function(self, tree, _body)
|
||||||
|
table.insert(tests, _body)
|
||||||
|
return LuaCode("")
|
||||||
|
end
|
||||||
|
end
|
||||||
local get_file_and_source
|
local get_file_and_source
|
||||||
get_file_and_source = function(filename)
|
get_file_and_source = function(filename)
|
||||||
local file, source
|
local file, source
|
||||||
@ -188,6 +198,7 @@ run = function()
|
|||||||
tree
|
tree
|
||||||
}
|
}
|
||||||
end
|
end
|
||||||
|
tests = { }
|
||||||
for _index_0 = 1, #tree do
|
for _index_0 = 1, #tree do
|
||||||
local chunk = tree[_index_0]
|
local chunk = tree[_index_0]
|
||||||
local lua = nomsu:compile(chunk):as_statements("return ")
|
local lua = nomsu:compile(chunk):as_statements("return ")
|
||||||
@ -198,6 +209,16 @@ run = function()
|
|||||||
end
|
end
|
||||||
nomsu:run_lua(lua)
|
nomsu:run_lua(lua)
|
||||||
end
|
end
|
||||||
|
if args.run_tests and #tests > 0 then
|
||||||
|
for _index_0 = 1, #tests do
|
||||||
|
local t = tests[_index_0]
|
||||||
|
local lua = nomsu:compile(t)
|
||||||
|
if lua_handler then
|
||||||
|
lua_handler(tostring(lua))
|
||||||
|
end
|
||||||
|
nomsu:run_lua(lua)
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
local parse_errs = { }
|
local parse_errs = { }
|
||||||
@ -294,9 +315,11 @@ say ".."
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
local has_ldt, ldt = pcall(require, 'ldt')
|
local debugger = require(args.debugger or 'error_handling')
|
||||||
if has_ldt then
|
local guard
|
||||||
return ldt.guard(run)
|
if type(debugger) == 'function' then
|
||||||
|
guard = debugger
|
||||||
else
|
else
|
||||||
return Errhand.run_safely(run)
|
guard = debugger.guard or debugger.call or debugger.wrap or debugger.run
|
||||||
end
|
end
|
||||||
|
return guard(run)
|
||||||
|
26
nomsu.moon
26
nomsu.moon
@ -11,13 +11,14 @@ EXIT_SUCCESS, EXIT_FAILURE = 0, 1
|
|||||||
usage = [=[
|
usage = [=[
|
||||||
Nomsu Compiler
|
Nomsu Compiler
|
||||||
|
|
||||||
Usage: (nomsu | lua nomsu.lua | moon nomsu.moon) [-V version] [-O] [-v] [-c] [-s] [-I file] [--help | -h] [--version] [file [nomsu args...]]
|
Usage: (nomsu | lua nomsu.lua | moon nomsu.moon) [-V version] [-O] [-v] [-c] [-s] [-t] [-I file] [--help | -h] [--version] [file [nomsu args...]]
|
||||||
|
|
||||||
OPTIONS
|
OPTIONS
|
||||||
-O Run the compiler in optimized mode (use precompiled .lua versions of Nomsu files, when available).
|
-O Run the compiler in optimized mode (use precompiled .lua versions of Nomsu files, when available).
|
||||||
-v Verbose: print compiled lua code.
|
-v Verbose: print compiled lua code.
|
||||||
-c Compile the input files into a .lua files.
|
-c Compile the input files into a .lua files.
|
||||||
-s Check the input files for syntax errors.
|
-s Check the input files for syntax errors.
|
||||||
|
-t Run tests.
|
||||||
-I <file> Add an additional input file or directory.
|
-I <file> Add an additional input file or directory.
|
||||||
-h/--help Print this message.
|
-h/--help Print this message.
|
||||||
--version Print the version number and exit.
|
--version Print the version number and exit.
|
||||||
@ -49,9 +50,11 @@ parser = re.compile([[
|
|||||||
/ ("-I" (";")? ({~ file ~} -> add_file))
|
/ ("-I" (";")? ({~ file ~} -> add_file))
|
||||||
/ ({:check_syntax: ("-s" -> true):})
|
/ ({:check_syntax: ("-s" -> true):})
|
||||||
/ ({:compile: ("-c" -> true):})
|
/ ({:compile: ("-c" -> true):})
|
||||||
|
/ {:run_tests: ("-t" -> true) :}
|
||||||
/ {:verbose: ("-v" -> true) :}
|
/ {:verbose: ("-v" -> true) :}
|
||||||
/ {:help: (("-h" / "--help") -> true) :}
|
/ {:help: (("-h" / "--help") -> true) :}
|
||||||
/ {:version: ("--version" -> true) :}
|
/ {:version: ("--version" -> true) :}
|
||||||
|
/ {:debugger: ("-d" (";")? {([^;])*}) :}
|
||||||
/ {:requested_version: "-V" ((";")? {([0-9.])+})? :}
|
/ {:requested_version: "-V" ((";")? {([0-9.])+})? :}
|
||||||
file <- ("-" -> "stdin") / {[^;]+}
|
file <- ("-" -> "stdin") / {[^;]+}
|
||||||
]], {
|
]], {
|
||||||
@ -99,6 +102,12 @@ run = ->
|
|||||||
return false if args.compile and input_files[f]
|
return false if args.compile and input_files[f]
|
||||||
return true
|
return true
|
||||||
|
|
||||||
|
tests = {}
|
||||||
|
if args.run_tests
|
||||||
|
nomsu.COMPILE_ACTIONS["test %"] = (tree, _body)=>
|
||||||
|
table.insert tests, _body
|
||||||
|
return LuaCode ""
|
||||||
|
|
||||||
get_file_and_source = (filename)->
|
get_file_and_source = (filename)->
|
||||||
local file, source
|
local file, source
|
||||||
if filename == 'stdin'
|
if filename == 'stdin'
|
||||||
@ -124,12 +133,18 @@ run = ->
|
|||||||
-- Each chunk's compilation is affected by the code in the previous chunks
|
-- Each chunk's compilation is affected by the code in the previous chunks
|
||||||
-- (typically), so each chunk needs to compile and run before the next one
|
-- (typically), so each chunk needs to compile and run before the next one
|
||||||
-- compiles.
|
-- compiles.
|
||||||
|
tests = {}
|
||||||
for chunk in *tree
|
for chunk in *tree
|
||||||
lua = nomsu\compile(chunk)\as_statements("return ")
|
lua = nomsu\compile(chunk)\as_statements("return ")
|
||||||
lua\declare_locals!
|
lua\declare_locals!
|
||||||
lua\prepend "-- File: #{source.filename\gsub("\n.*", "...")}\n"
|
lua\prepend "-- File: #{source.filename\gsub("\n.*", "...")}\n"
|
||||||
if lua_handler then lua_handler(tostring(lua))
|
if lua_handler then lua_handler(tostring(lua))
|
||||||
nomsu\run_lua(lua)
|
nomsu\run_lua(lua)
|
||||||
|
if args.run_tests and #tests > 0
|
||||||
|
for t in *tests
|
||||||
|
lua = nomsu\compile(t)
|
||||||
|
if lua_handler then lua_handler(tostring(lua))
|
||||||
|
nomsu\run_lua(lua)
|
||||||
|
|
||||||
parse_errs = {}
|
parse_errs = {}
|
||||||
for f in *file_queue
|
for f in *file_queue
|
||||||
@ -201,8 +216,7 @@ say ".."
|
|||||||
elseif not ok
|
elseif not ok
|
||||||
Errhand.print_error ret
|
Errhand.print_error ret
|
||||||
|
|
||||||
has_ldt, ldt = pcall(require,'ldt')
|
debugger = require(args.debugger or 'error_handling')
|
||||||
if has_ldt
|
guard = if type(debugger) == 'function' then debugger
|
||||||
ldt.guard(run)
|
else debugger.guard or debugger.call or debugger.wrap or debugger.run
|
||||||
else
|
guard(run)
|
||||||
Errhand.run_safely(run)
|
|
||||||
|
@ -301,13 +301,13 @@ do
|
|||||||
return lua
|
return lua
|
||||||
end,
|
end,
|
||||||
["Lua %"] = function(self, tree, _code)
|
["Lua %"] = function(self, tree, _code)
|
||||||
local lua = LuaCode.Value(_code.source, "LuaCode(", repr(tostring(_code.source)))
|
local lua = LuaCode.Value(tree.source, "LuaCode(", repr(tostring(_code.source)))
|
||||||
add_lua_string_bits(self, lua, _code)
|
add_lua_string_bits(self, lua, _code)
|
||||||
lua:append(")")
|
lua:append(")")
|
||||||
return lua
|
return lua
|
||||||
end,
|
end,
|
||||||
["Lua value %"] = function(self, tree, _code)
|
["Lua value %"] = function(self, tree, _code)
|
||||||
local lua = LuaCode.Value(_code.source, "LuaCode.Value(", repr(tostring(_code.source)))
|
local lua = LuaCode.Value(tree.source, "LuaCode.Value(", repr(tostring(_code.source)))
|
||||||
add_lua_string_bits(self, lua, _code)
|
add_lua_string_bits(self, lua, _code)
|
||||||
lua:append(")")
|
lua:append(")")
|
||||||
return lua
|
return lua
|
||||||
@ -332,6 +332,9 @@ do
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
return LuaCode(tree.source, "for f in files.walk(", self:compile(_path), ") do nomsu:run_file(f) end")
|
return LuaCode(tree.source, "for f in files.walk(", self:compile(_path), ") do nomsu:run_file(f) end")
|
||||||
|
end,
|
||||||
|
["test %"] = function(self, tree, _body)
|
||||||
|
return LuaCode("")
|
||||||
end
|
end
|
||||||
}, {
|
}, {
|
||||||
__index = function(self, stub)
|
__index = function(self, stub)
|
||||||
@ -434,7 +437,7 @@ do
|
|||||||
source = nil
|
source = nil
|
||||||
end
|
end
|
||||||
local lua_string = tostring(lua)
|
local lua_string = tostring(lua)
|
||||||
local run_lua_fn, err = load(lua_string, nil and tostring(source or lua.source), "t", self)
|
local run_lua_fn, err = load(lua_string, tostring(source or lua.source), "t", self)
|
||||||
if not run_lua_fn then
|
if not run_lua_fn then
|
||||||
local line_numbered_lua = concat((function()
|
local line_numbered_lua = concat((function()
|
||||||
local _accum_0 = { }
|
local _accum_0 = { }
|
||||||
@ -447,39 +450,36 @@ do
|
|||||||
end)(), "\n")
|
end)(), "\n")
|
||||||
error("Failed to compile generated code:\n" .. tostring(colored.bright(colored.blue(colored.onblack(line_numbered_lua)))) .. "\n\n" .. tostring(err), 0)
|
error("Failed to compile generated code:\n" .. tostring(colored.bright(colored.blue(colored.onblack(line_numbered_lua)))) .. "\n\n" .. tostring(err), 0)
|
||||||
end
|
end
|
||||||
local source_key = tostring(source or lua.source)
|
source = source or lua.source
|
||||||
|
local source_key = tostring(source)
|
||||||
if not (SOURCE_MAP[source_key]) then
|
if not (SOURCE_MAP[source_key]) then
|
||||||
local map = { }
|
local map = { }
|
||||||
local offset = 1
|
|
||||||
source = source or lua.source
|
|
||||||
local file = files.read(source.filename)
|
local file = files.read(source.filename)
|
||||||
if not file then
|
if not file then
|
||||||
error("Failed to find file: " .. tostring(source.filename))
|
error("Failed to find file: " .. tostring(source.filename))
|
||||||
end
|
end
|
||||||
local nomsu_str = tostring(file:sub(source.start, source.stop))
|
local nomsu_str = tostring(file:sub(source.start, source.stop))
|
||||||
local lua_line = 1
|
local lua_line = 1
|
||||||
local nomsu_line = files.get_line_number(nomsu_str, source.start)
|
local nomsu_line = files.get_line_number(file, source.start)
|
||||||
local fn
|
local map_sources
|
||||||
fn = function(s)
|
map_sources = function(s)
|
||||||
if type(s) == 'string' then
|
if type(s) == 'string' then
|
||||||
for nl in s:gmatch("\n") do
|
for nl in s:gmatch("\n") do
|
||||||
map[lua_line] = map[lua_line] or nomsu_line
|
map[lua_line] = map[lua_line] or nomsu_line
|
||||||
lua_line = lua_line + 1
|
lua_line = lua_line + 1
|
||||||
end
|
end
|
||||||
else
|
else
|
||||||
local old_line = nomsu_line
|
if s.source and s.source.filename == source.filename then
|
||||||
if s.source then
|
nomsu_line = files.get_line_number(file, s.source.start)
|
||||||
nomsu_line = files.get_line_number(nomsu_str, s.source.start)
|
|
||||||
end
|
end
|
||||||
local _list_0 = s.bits
|
local _list_0 = s.bits
|
||||||
for _index_0 = 1, #_list_0 do
|
for _index_0 = 1, #_list_0 do
|
||||||
local b = _list_0[_index_0]
|
local b = _list_0[_index_0]
|
||||||
fn(b)
|
map_sources(b)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
fn(lua)
|
map_sources(lua)
|
||||||
map[lua_line] = map[lua_line] or nomsu_line
|
|
||||||
map[0] = 0
|
map[0] = 0
|
||||||
SOURCE_MAP[source_key] = map
|
SOURCE_MAP[source_key] = map
|
||||||
end
|
end
|
||||||
|
@ -189,13 +189,13 @@ with NomsuCompiler
|
|||||||
return lua
|
return lua
|
||||||
|
|
||||||
["Lua %"]: (tree, _code)=>
|
["Lua %"]: (tree, _code)=>
|
||||||
lua = LuaCode.Value(_code.source, "LuaCode(", repr(tostring _code.source))
|
lua = LuaCode.Value(tree.source, "LuaCode(", repr(tostring _code.source))
|
||||||
add_lua_string_bits(@, lua, _code)
|
add_lua_string_bits(@, lua, _code)
|
||||||
lua\append ")"
|
lua\append ")"
|
||||||
return lua
|
return lua
|
||||||
|
|
||||||
["Lua value %"]: (tree, _code)=>
|
["Lua value %"]: (tree, _code)=>
|
||||||
lua = LuaCode.Value(_code.source, "LuaCode.Value(", repr(tostring _code.source))
|
lua = LuaCode.Value(tree.source, "LuaCode.Value(", repr(tostring _code.source))
|
||||||
add_lua_string_bits(@, lua, _code)
|
add_lua_string_bits(@, lua, _code)
|
||||||
lua\append ")"
|
lua\append ")"
|
||||||
return lua
|
return lua
|
||||||
@ -216,6 +216,9 @@ with NomsuCompiler
|
|||||||
for f in files.walk(path)
|
for f in files.walk(path)
|
||||||
@run_file(f)
|
@run_file(f)
|
||||||
return LuaCode(tree.source, "for f in files.walk(", @compile(_path), ") do nomsu:run_file(f) end")
|
return LuaCode(tree.source, "for f in files.walk(", @compile(_path), ") do nomsu:run_file(f) end")
|
||||||
|
|
||||||
|
["test %"]: (tree, _body)=>
|
||||||
|
return LuaCode ""
|
||||||
}, {
|
}, {
|
||||||
__index: (stub)=>
|
__index: (stub)=>
|
||||||
if math_expression\match(stub)
|
if math_expression\match(stub)
|
||||||
@ -285,35 +288,33 @@ with NomsuCompiler
|
|||||||
|
|
||||||
.run_lua = (lua, source=nil)=>
|
.run_lua = (lua, source=nil)=>
|
||||||
lua_string = tostring(lua)
|
lua_string = tostring(lua)
|
||||||
run_lua_fn, err = load(lua_string, nil and tostring(source or lua.source), "t", self)
|
run_lua_fn, err = load(lua_string, tostring(source or lua.source), "t", self)
|
||||||
if not run_lua_fn
|
if not run_lua_fn
|
||||||
line_numbered_lua = concat(
|
line_numbered_lua = concat(
|
||||||
[format("%3d|%s",i,line) for i, line in ipairs files.get_lines(lua_string)],
|
[format("%3d|%s",i,line) for i, line in ipairs files.get_lines(lua_string)],
|
||||||
"\n")
|
"\n")
|
||||||
error("Failed to compile generated code:\n#{colored.bright colored.blue colored.onblack line_numbered_lua}\n\n#{err}", 0)
|
error("Failed to compile generated code:\n#{colored.bright colored.blue colored.onblack line_numbered_lua}\n\n#{err}", 0)
|
||||||
source_key = tostring(source or lua.source)
|
source or= lua.source
|
||||||
|
source_key = tostring(source)
|
||||||
unless SOURCE_MAP[source_key]
|
unless SOURCE_MAP[source_key]
|
||||||
map = {}
|
map = {}
|
||||||
offset = 1
|
|
||||||
source or= lua.source
|
|
||||||
file = files.read(source.filename)
|
file = files.read(source.filename)
|
||||||
if not file
|
if not file
|
||||||
error "Failed to find file: #{source.filename}"
|
error "Failed to find file: #{source.filename}"
|
||||||
nomsu_str = tostring(file\sub(source.start, source.stop))
|
nomsu_str = tostring(file\sub(source.start, source.stop))
|
||||||
lua_line = 1
|
lua_line = 1
|
||||||
nomsu_line = files.get_line_number(nomsu_str, source.start)
|
nomsu_line = files.get_line_number(file, source.start)
|
||||||
fn = (s)->
|
map_sources = (s)->
|
||||||
if type(s) == 'string'
|
if type(s) == 'string'
|
||||||
for nl in s\gmatch("\n")
|
for nl in s\gmatch("\n")
|
||||||
map[lua_line] or= nomsu_line
|
map[lua_line] or= nomsu_line
|
||||||
lua_line += 1
|
lua_line += 1
|
||||||
else
|
else
|
||||||
old_line = nomsu_line
|
if s.source and s.source.filename == source.filename
|
||||||
if s.source
|
nomsu_line = files.get_line_number(file, s.source.start)
|
||||||
nomsu_line = files.get_line_number(nomsu_str, s.source.start)
|
for b in *s.bits do map_sources(b)
|
||||||
for b in *s.bits do fn(b)
|
map_sources(lua)
|
||||||
fn(lua)
|
--map[lua_line] or= nomsu_line
|
||||||
map[lua_line] or= nomsu_line
|
|
||||||
map[0] = 0
|
map[0] = 0
|
||||||
-- Mapping from lua line number to nomsu line numbers
|
-- Mapping from lua line number to nomsu line numbers
|
||||||
SOURCE_MAP[source_key] = map
|
SOURCE_MAP[source_key] = map
|
||||||
|
Loading…
Reference in New Issue
Block a user