2018-06-19 01:12:43 -07:00
local debug_getinfo = debug.getinfo
2019-01-14 15:42:48 -08:00
local Files = require ( " files " )
2019-01-29 16:16:45 -08:00
local C = require ( " colors " )
2019-01-18 20:46:04 -08:00
local pretty_error = require ( " pretty_errors " )
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
2019-01-19 18:27:30 -08:00
info.name = " action ' " .. tostring ( info.name : from_lua_id ( ) ) .. " ' "
2018-07-10 15:00:01 -07:00
else
info.name = " main chunk "
end
2018-06-19 01:12:43 -07:00
end
end
end
return info
end
2019-02-02 19:42:14 -08:00
local strdist
strdist = function ( a , b , cache )
if cache == nil then
cache = { }
end
if a == b then
return 0
end
if # a < # b then
a , b = b , a
end
if b == " " then
return # a
end
local k = a .. ' \003 ' .. b
if not ( cache [ k ] ) then
cache [ k ] = math.min ( strdist ( a : sub ( 1 , - 2 ) , b , cache ) + 1 , strdist ( a , b : sub ( 1 , - 2 ) , cache ) + 1 , strdist ( a : sub ( 1 , - 2 ) , b : sub ( 1 , - 2 ) , cache ) + ( a : sub ( - 1 ) ~= b : sub ( - 1 ) and 1.1 or 0 ) )
if # a >= 2 and # b >= 2 and a : sub ( - 1 , - 1 ) == b : sub ( - 2 , - 2 ) and a : sub ( - 2 , - 2 ) == b : sub ( - 1 , - 1 ) then
cache [ k ] = math.min ( cache [ k ] , strdist ( a : sub ( 1 , - 3 ) , b : sub ( 1 , - 3 ) , cache ) + 1 )
end
end
return cache [ k ]
end
2019-01-18 20:46:04 -08:00
local enhance_error
2019-02-02 19:42:14 -08:00
enhance_error = function ( error_message )
2019-01-29 16:16:45 -08:00
if not ( error_message and error_message : match ( " %d| " ) ) then
2019-01-18 20:46:04 -08:00
error_message = error_message or " "
do
2019-03-04 14:19:44 -08:00
local fn_name = error_message : match ( " attempt to call a nil value %(method '(.*)'%) " )
2019-02-02 19:42:14 -08:00
if fn_name then
local action_name = fn_name : from_lua_id ( )
2019-03-04 14:19:44 -08:00
error_message = " This object does not have the method ' " .. tostring ( action_name ) .. " '. "
else
do
fn_name = ( error_message : match ( " attempt to call a nil value %(global '(.*)'%) " ) or error_message : match ( " attempt to call global '(.*)' %(a nil value%) " ) )
if fn_name then
local action_name = fn_name : from_lua_id ( )
error_message = " The action ' " .. tostring ( action_name ) .. " ' is not defined. "
local func = debug.getinfo ( 2 , ' f ' ) . func
2019-03-14 17:57:30 -07:00
local env
if _VERSION == " Lua 5.1 " then
env = getfenv ( func )
else
local ename
2019-03-04 14:19:44 -08:00
ename , env = debug.getupvalue ( func , 1 )
2019-03-14 17:57:30 -07:00
if not ( ename == " _ENV " or ename == " _G " ) then
func = debug.getinfo ( 3 , ' f ' ) . func
ename , env = debug.getupvalue ( func , 1 )
end
2019-02-02 19:42:14 -08:00
end
2019-03-04 14:19:44 -08:00
local THRESHOLD = math.min ( 4.5 , .9 * # action_name )
local candidates = { }
local cache = { }
for i = 1 , 99 do
local k , v = debug.getlocal ( 2 , i )
if k == nil then
break
end
if not ( k : sub ( 1 , 1 ) == " ( " or type ( v ) ~= ' function ' ) then
k = k : from_lua_id ( )
if strdist ( k , action_name , cache ) <= THRESHOLD and k ~= " " then
table.insert ( candidates , k )
end
end
2019-02-02 19:42:14 -08:00
end
2019-03-04 14:19:44 -08:00
for i = 1 , debug.getinfo ( func , ' u ' ) . nups do
local k , v = debug.getupvalue ( func , i )
if not ( k : sub ( 1 , 1 ) == " ( " or type ( v ) ~= ' function ' ) then
2019-02-02 19:42:14 -08:00
k = k : from_lua_id ( )
2019-03-04 14:19:44 -08:00
if strdist ( k , action_name , cache ) <= THRESHOLD and k ~= " " then
table.insert ( candidates , k )
end
2019-02-02 19:42:14 -08:00
end
2019-03-04 14:19:44 -08:00
end
local scan
scan = function ( t , is_lua_id )
2019-03-27 14:37:38 -07:00
if not ( t ) then
return
end
2019-03-04 14:19:44 -08:00
for k , v in pairs ( t ) do
if type ( k ) == ' string ' and type ( v ) == ' function ' then
if not ( is_lua_id ) then
k = k : from_lua_id ( )
end
if strdist ( k , action_name , cache ) <= THRESHOLD and k ~= " " then
table.insert ( candidates , k )
end
end
2019-02-02 19:42:14 -08:00
end
end
2019-03-04 14:19:44 -08:00
scan ( env.COMPILE_RULES , true )
scan ( env.COMPILE_RULES . _IMPORTS , true )
scan ( env )
scan ( env._IMPORTS )
if # candidates > 0 then
for _index_0 = 1 , # candidates do
local c = candidates [ _index_0 ]
THRESHOLD = math.min ( THRESHOLD , strdist ( c , action_name , cache ) )
end
do
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1 , # candidates do
local c = candidates [ _index_0 ]
if strdist ( c , action_name , cache ) <= THRESHOLD then
_accum_0 [ _len_0 ] = c
_len_0 = _len_0 + 1
end
end
candidates = _accum_0
end
if # candidates == 1 then
error_message = error_message .. " \n \x1b [3mSuggestion: Maybe you meant ' " .. tostring ( candidates [ 1 ] ) .. " '? "
elseif # candidates > 0 then
local last = table.remove ( candidates )
error_message = error_message .. ( " \n " .. C ( ' italic ' , " Suggestion: Maybe you meant ' " .. tostring ( table.concat ( candidates , " ', ' " ) ) .. " ' " .. tostring ( # candidates > 1 and ' , ' or ' ' ) .. " or ' " .. tostring ( last ) .. " '? " ) )
2019-02-02 19:42:14 -08:00
end
end
end
end
2019-01-18 20:46:04 -08:00
end
end
local level = 2
while true do
local calling_fn = debug_getinfo ( level )
if not calling_fn then
break
end
level = level + 1
local filename , file , line_num
do
local map = SOURCE_MAP and SOURCE_MAP [ calling_fn.source ]
if map then
if calling_fn.currentline then
line_num = assert ( map [ calling_fn.currentline ] )
end
local start , stop
filename , start , stop = calling_fn.source : match ( ' @([^[]*)%[([0-9]+):([0-9]+)] ' )
if not filename then
filename , start = calling_fn.source : match ( ' @([^[]*)%[([0-9]+)] ' )
end
assert ( filename )
file = Files.read ( filename )
else
filename = calling_fn.short_src
file = Files.read ( filename )
if calling_fn.short_src : match ( " %.moon$ " ) and type ( MOON_SOURCE_MAP [ file ] ) == ' table ' then
local char = MOON_SOURCE_MAP [ file ] [ calling_fn.currentline ]
line_num = file : line_number_at ( char )
else
line_num = calling_fn.currentline
end
end
end
if file and filename and line_num then
local start = 1
local lines = file : lines ( )
for i = 1 , line_num - 1 do
start = start + # lines [ i ] + 1
end
local stop = start + # lines [ line_num ]
start = start + # lines [ line_num ] : match ( " ^ * " )
error_message = pretty_error ( {
title = " Error " ,
error = error_message ,
source = file ,
start = start ,
stop = stop ,
filename = filename
} )
break
end
if calling_fn.func == xpcall then
break
end
end
end
local ret = {
2019-01-29 16:16:45 -08:00
C ( ' bold red ' , error_message or " Error " ) ,
2019-01-18 20:46:04 -08:00
" stack traceback: "
}
local level = 2
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
2019-01-18 20:46:04 -08:00
if calling_fn.func == xpcall then
2018-06-19 01:12:43 -07:00
break
end
2019-01-18 20:46:04 -08:00
level = level + 1
2018-06-19 01:12:43 -07:00
local name = calling_fn.name and " function ' " .. tostring ( calling_fn.name ) .. " ' " or nil
if calling_fn.linedefined == 0 then
name = " main chunk "
end
2019-01-18 20:46:04 -08:00
if name == " function 'run_lua_fn' " then
2018-06-19 01:12:43 -07:00
_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-06-19 02:00:52 -07:00
if calling_fn.name then
2019-01-18 20:46:04 -08:00
name = " action ' " .. tostring ( calling_fn.name : from_lua_id ( ) ) .. " ' "
2018-06-19 02:00:52 -07:00
else
name = " main chunk "
end
2019-01-14 15:42:48 -08:00
local file = Files.read ( filename )
local lines = file and file : lines ( ) or { }
2018-07-23 15:25:53 -07:00
do
2019-01-14 15:42:48 -08:00
local err_line = lines [ calling_fn.currentline ]
2018-07-23 15:25:53 -07:00
if err_line then
2019-01-29 16:16:45 -08:00
local offending_statement = C ( ' bright red ' , err_line : match ( " ^[ ]*(.*) " ) )
line = C ( ' yellow ' , tostring ( filename ) .. " : " .. tostring ( calling_fn.currentline ) .. " in " .. tostring ( name ) .. " \n " .. tostring ( offending_statement ) )
2018-07-23 15:25:53 -07:00
else
2019-01-29 16:16:45 -08:00
line = C ( ' yellow ' , tostring ( filename ) .. " : " .. tostring ( calling_fn.currentline ) .. " in " .. tostring ( name ) )
2018-07-23 15:25:53 -07:00
end
end
2018-06-19 01:12:43 -07:00
else
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
2019-01-14 15:42:48 -08:00
local file , lines
do
file = Files.read ( calling_fn.short_src )
if file then
lines = file : lines ( )
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 ]
2019-01-18 20:46:04 -08:00
line_num = file : line_number_at ( char )
2019-01-29 16:16:45 -08:00
line = C ( ' cyan ' , tostring ( calling_fn.short_src ) .. " : " .. tostring ( line_num ) .. " in " .. tostring ( name or ' ? ' ) )
2018-06-19 01:12:43 -07:00
else
line_num = calling_fn.currentline
if calling_fn.short_src == ' [C] ' then
2019-01-29 16:16:45 -08:00
line = C ( ' green ' , tostring ( calling_fn.short_src ) .. " in " .. tostring ( name or ' ? ' ) )
2018-06-19 01:12:43 -07:00
else
2019-01-29 16:16:45 -08:00
line = C ( ' blue ' , tostring ( calling_fn.short_src ) .. " : " .. tostring ( calling_fn.currentline ) .. " in " .. tostring ( name or ' ? ' ) )
2018-06-19 01:12:43 -07:00
end
end
if file then
2018-07-23 15:25:53 -07:00
do
2019-01-14 15:42:48 -08:00
local err_line = lines [ line_num ]
2018-07-23 15:25:53 -07:00
if err_line then
2019-01-29 16:16:45 -08:00
local offending_statement = C ( ' bright red ' , tostring ( err_line : match ( " ^[ ]*(.*)$ " ) ) )
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
2019-01-18 20:46:04 -08:00
table.insert ( ret , line )
2018-06-19 01:12:43 -07:00
if calling_fn.istailcall then
2019-01-29 16:16:45 -08:00
table.insert ( ret , C ( ' dim ' , " (...tail calls...) " ) )
2018-07-10 15:00:01 -07:00
end
2018-06-19 01:12:43 -07:00
_continue_0 = true
until true
if not _continue_0 then
break
end
end
2019-01-18 20:46:04 -08:00
return table.concat ( ret , " \n " )
2018-06-19 01:12:43 -07:00
end
2018-07-10 15:00:01 -07:00
local guard
guard = function ( fn )
2019-01-18 20:46:04 -08:00
local err
ok , err = xpcall ( fn , enhance_error )
if not ok then
io.stderr : write ( err )
io.stderr : flush ( )
return os.exit ( 1 )
2018-07-10 15:00:01 -07:00
end
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 ,
2019-01-18 20:46:04 -08:00
enhance_error = enhance_error ,
2018-07-10 15:00:01 -07:00
print_error = print_error
2018-06-19 15:24:24 -07:00
}