nomsu/error_handling.lua

248 lines
8.2 KiB
Lua
Raw Normal View History

2018-06-19 01:12:43 -07:00
local debug_getinfo = debug.getinfo
local Files = require("files")
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
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
print_error = function(error_message, start_fn, stop_fn)
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")
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
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
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]+)]')
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
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
local file = Files.read(filename)
local lines = file and file:lines() or { }
2018-07-23 15:25:53 -07:00
do
local err_line = lines[calling_fn.currentline]
2018-07-23 15:25:53 -07:00
if err_line then
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
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)
while true do
2018-06-19 01:12:43 -07:00
search_level = search_level + 1
_info = debug.getinfo(search_level)
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
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]
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
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
line = tostring(GREEN) .. tostring(calling_fn.short_src) .. " in " .. tostring(name or '?') .. tostring(RESET)
2018-06-19 01:12:43 -07:00
else
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 = lines[line_num]
2018-07-23 15:25:53 -07:00
if err_line then
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
io.stderr:write(line, "\n")
2018-06-19 01:12:43 -07:00
if calling_fn.istailcall then
io.stderr:write(" " .. tostring(DIM) .. "(...tail calls...)" .. tostring(RESET) .. "\n")
2018-06-19 01:12:43 -07:00
end
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
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 {
guard = guard,
print_error = print_error
2018-06-19 15:24:24 -07:00
}