2018-06-23 17:22:23 -07:00
local files = require ( " files " )
2018-06-19 01:12:43 -07:00
local debug_getinfo = debug.getinfo
2018-11-08 15:23:22 -08:00
local RED = " \027 [31m "
local BRIGHT_RED = " \027 [31;1m "
local RESET = " \027 [0m "
local YELLOW = " \027 [33m "
local CYAN = " \027 [36m "
local GREEN = " \027 [32m "
local BLUE = " \027 [34m "
local DIM = " \027 [37;2m "
2018-06-19 01:12:43 -07:00
local ok , to_lua = pcall ( function ( )
return require ( ' moonscript.base ' ) . to_lua
end )
if not ok then
2018-06-19 02:00:52 -07:00
to_lua = function ( )
return nil
end
2018-06-19 01:12:43 -07:00
end
2018-06-19 02:00:52 -07:00
local MOON_SOURCE_MAP = setmetatable ( { } , {
__index = function ( self , file )
local _ , line_table = to_lua ( file )
self [ file ] = line_table or false
return line_table or false
2018-06-19 01:12:43 -07:00
end
} )
debug.getinfo = function ( thread , f , what )
if what == nil then
f , what , thread = thread , f , nil
end
if type ( f ) == ' number ' then
f = f + 1
end
local info
if thread == nil then
info = debug_getinfo ( f , what )
else
info = debug_getinfo ( thread , f , what )
end
if not info or not info.func then
return info
end
if info.short_src or info.source or info.linedefine or info.currentline then
do
local map = SOURCE_MAP [ info.source ]
if map then
if info.currentline then
info.currentline = assert ( map [ info.currentline ] )
end
if info.linedefined then
info.linedefined = assert ( map [ info.linedefined ] )
end
if info.lastlinedefined then
info.lastlinedefined = assert ( map [ info.lastlinedefined ] )
end
info.short_src = info.source : match ( ' @([^[]*) ' ) or info.short_src
2018-07-10 15:00:01 -07:00
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
2018-06-19 01:12:43 -07:00
end
end
end
return info
end
2018-06-19 15:24:24 -07:00
local print_error
2018-07-10 15:00:01 -07:00
print_error = function ( error_message , start_fn , stop_fn )
2018-11-08 15:23:22 -08:00
io.stderr : write ( tostring ( RED ) .. " ERROR: " .. tostring ( BRIGHT_RED ) .. tostring ( error_message or " " ) .. tostring ( RESET ) .. " \n " )
2018-06-19 01:12:43 -07:00
io.stderr : write ( " stack traceback: \n " )
2018-07-10 15:00:01 -07:00
local level = 1
local found_start = false
2018-06-19 01:12:43 -07:00
while true do
local _continue_0 = false
repeat
local calling_fn = debug_getinfo ( level )
if not calling_fn then
break
end
2018-07-10 15:00:01 -07:00
level = level + 1
if not ( found_start ) then
if calling_fn.func == start_fn then
found_start = true
end
_continue_0 = true
2018-06-19 01:12:43 -07:00
break
end
local name = calling_fn.name and " function ' " .. tostring ( calling_fn.name ) .. " ' " or nil
if calling_fn.linedefined == 0 then
name = " main chunk "
end
if name == " run_lua_fn " then
_continue_0 = true
break
end
local line = nil
do
2019-01-08 16:35:32 -08:00
local map = SOURCE_MAP and SOURCE_MAP [ calling_fn.source ]
2018-06-19 01:12:43 -07:00
if map then
if calling_fn.currentline then
calling_fn.currentline = assert ( map [ calling_fn.currentline ] )
end
if calling_fn.linedefined then
calling_fn.linedefined = assert ( map [ calling_fn.linedefined ] )
end
if calling_fn.lastlinedefined then
calling_fn.lastlinedefined = assert ( map [ calling_fn.lastlinedefined ] )
end
local filename , start , stop = calling_fn.source : match ( ' @([^[]*)%[([0-9]+):([0-9]+)] ' )
2018-12-18 17:35:27 -08:00
if not filename then
filename , start = calling_fn.source : match ( ' @([^[]*)%[([0-9]+)] ' )
end
2018-06-19 01:12:43 -07:00
assert ( filename )
2018-07-10 15:00:01 -07:00
local file = files.read ( filename )
2018-06-19 02:00:52 -07:00
if calling_fn.name then
do
local tmp = calling_fn.name : match ( " ^A_([a-zA-Z0-9_]*)$ " )
if tmp then
name = " action ' " .. tostring ( tmp : gsub ( " _ " , " " ) : gsub ( " x([0-9A-F][0-9A-F]) " , function ( self )
return string.char ( tonumber ( self , 16 ) )
end ) ) .. " ' "
else
name = " action ' " .. tostring ( calling_fn.name ) .. " ' "
end
end
else
name = " main chunk "
end
2018-07-23 15:25:53 -07:00
do
local err_line = files.get_line ( file , calling_fn.currentline )
if err_line then
2018-11-08 15:23:22 -08:00
local offending_statement = tostring ( BRIGHT_RED ) .. tostring ( err_line : match ( " ^[ ]*(.*) " ) ) .. tostring ( RESET )
line = tostring ( YELLOW ) .. tostring ( filename ) .. " : " .. tostring ( calling_fn.currentline ) .. " in " .. tostring ( name ) .. " \n " .. tostring ( offending_statement ) .. tostring ( RESET )
2018-07-23 15:25:53 -07:00
else
2018-11-08 15:23:22 -08:00
line = tostring ( YELLOW ) .. tostring ( filename ) .. " : " .. tostring ( calling_fn.currentline ) .. " in " .. tostring ( name ) .. tostring ( RESET )
2018-07-23 15:25:53 -07:00
end
end
2018-06-19 01:12:43 -07:00
else
local file
ok , file = pcall ( function ( )
2018-07-10 15:00:01 -07:00
return files.read ( calling_fn.short_src )
2018-06-19 01:12:43 -07:00
end )
if not ok then
file = nil
end
local line_num
if name == nil then
local search_level = level
local _info = debug.getinfo ( search_level )
2018-07-10 15:00:01 -07:00
while true do
2018-06-19 01:12:43 -07:00
search_level = search_level + 1
_info = debug.getinfo ( search_level )
2018-07-10 15:00:01 -07:00
if not ( _info ) then
break
end
2018-06-19 01:12:43 -07:00
for i = 1 , 999 do
local varname , val = debug.getlocal ( search_level , i )
if not varname then
break
end
if val == calling_fn.func then
name = " local ' " .. tostring ( varname ) .. " ' "
if not varname : match ( " %( " ) then
break
end
end
end
if not ( name ) then
for i = 1 , _info.nups do
local varname , val = debug.getupvalue ( _info.func , i )
if not varname then
break
end
if val == calling_fn.func then
name = " upvalue ' " .. tostring ( varname ) .. " ' "
if not varname : match ( " %( " ) then
break
end
end
end
end
end
end
2018-06-20 15:22:03 -07:00
if file and ( calling_fn.short_src : match ( " %.moon$ " ) or file : match ( " ^#![^ \n ]*moon \n " ) ) and type ( MOON_SOURCE_MAP [ file ] ) == ' table ' then
2018-06-19 02:00:52 -07:00
local char = MOON_SOURCE_MAP [ file ] [ calling_fn.currentline ]
2018-06-19 01:12:43 -07:00
line_num = 1
for _ in file : sub ( 1 , char ) : gmatch ( " \n " ) do
line_num = line_num + 1
end
2018-11-08 15:23:22 -08:00
line = tostring ( CYAN ) .. tostring ( calling_fn.short_src ) .. " : " .. tostring ( line_num ) .. " in " .. tostring ( name or ' ? ' ) .. tostring ( RESET )
2018-06-19 01:12:43 -07:00
else
line_num = calling_fn.currentline
if calling_fn.short_src == ' [C] ' then
2018-11-08 15:23:22 -08:00
line = tostring ( GREEN ) .. tostring ( calling_fn.short_src ) .. " in " .. tostring ( name or ' ? ' ) .. tostring ( RESET )
2018-06-19 01:12:43 -07:00
else
2018-11-08 15:23:22 -08:00
line = tostring ( BLUE ) .. tostring ( calling_fn.short_src ) .. " : " .. tostring ( calling_fn.currentline ) .. " in " .. tostring ( name or ' ? ' ) .. tostring ( RESET )
2018-06-19 01:12:43 -07:00
end
end
if file then
2018-07-23 15:25:53 -07:00
do
local err_line = files.get_line ( file , line_num )
if err_line then
2018-11-08 15:23:22 -08:00
local offending_statement = tostring ( BRIGHT_RED ) .. tostring ( err_line : match ( " ^[ ]*(.*)$ " ) ) .. tostring ( RESET )
2018-07-23 15:25:53 -07:00
line = line .. ( " \n " .. offending_statement )
end
end
2018-06-19 01:12:43 -07:00
end
end
end
2018-07-10 15:00:01 -07:00
io.stderr : write ( line , " \n " )
2018-06-19 01:12:43 -07:00
if calling_fn.istailcall then
2018-11-08 15:23:22 -08:00
io.stderr : write ( " " .. tostring ( DIM ) .. " (...tail calls...) " .. tostring ( RESET ) .. " \n " )
2018-06-19 01:12:43 -07:00
end
2018-07-10 15:00:01 -07:00
if calling_fn.func == stop_fn then
break
end
2018-06-19 01:12:43 -07:00
_continue_0 = true
until true
if not _continue_0 then
break
end
end
return io.stderr : flush ( )
end
2018-07-10 15:00:01 -07:00
local guard
guard = function ( fn )
local error_handler
error_handler = function ( error_message )
print_error ( error_message , error_handler , fn )
local EXIT_FAILURE = 1
return os.exit ( EXIT_FAILURE )
end
2018-06-19 15:24:24 -07:00
return xpcall ( fn , error_handler )
2018-06-19 01:12:43 -07:00
end
2018-06-19 15:24:24 -07:00
return {
2018-07-10 15:00:01 -07:00
guard = guard ,
print_error = print_error
2018-06-19 15:24:24 -07:00
}