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:
Bruce Hill 2018-07-10 15:00:01 -07:00
parent de34592dbe
commit fa72d7eeb1
12 changed files with 212 additions and 102 deletions

View File

@ -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"

View File

@ -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("#",...)

View File

@ -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 [..]

View File

@ -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

View File

@ -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()"

View File

@ -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))"
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

View File

@ -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
} }

View File

@ -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}

View File

@ -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)

View File

@ -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)

View File

@ -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

View File

@ -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