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-18 20:46:04 -08:00
local pretty_error = require ( " pretty_errors " )
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
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-01-18 20:46:04 -08:00
local enhance_error
enhance_error = function ( error_message , start_fn , stop_fn )
if not ( error_message and error_message : match ( " \x1b " ) ) then
error_message = error_message or " "
do
local fn = error_message : match ( " attempt to call a nil value %(global '(.*)'%) " )
if fn then
if fn : match ( " x[0-9A-F][0-9A-F] " ) then
error_message = " The action ' " .. tostring ( fn : from_lua_id ( ) ) .. " ' is not defined. "
end
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 = {
tostring ( RED ) .. " ERROR: " .. tostring ( BRIGHT_RED ) .. tostring ( error_message or " " ) .. tostring ( RESET ) ,
" 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
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 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 )
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
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
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
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-18 20:46:04 -08:00
table.insert ( ret , " " .. tostring ( DIM ) .. " (...tail calls...) " .. tostring ( RESET ) )
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
}