Better error handling.

This commit is contained in:
Bruce Hill 2019-01-18 20:46:04 -08:00
parent 13cab23e20
commit 5a99a24176
5 changed files with 165 additions and 85 deletions

View File

@ -1,5 +1,6 @@
local debug_getinfo = debug.getinfo local debug_getinfo = debug.getinfo
local Files = require("files") local Files = require("files")
local pretty_error = require("pretty_errors")
local RED = "\027[31m" local RED = "\027[31m"
local BRIGHT_RED = "\027[31;1m" local BRIGHT_RED = "\027[31;1m"
local RESET = "\027[0m" local RESET = "\027[0m"
@ -54,16 +55,7 @@ debug.getinfo = function(thread, f, what)
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 if info.name then
do info.name = "action '" .. tostring(calling_fn.name:from_lua_id()) .. "'"
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 else
info.name = "main chunk" info.name = "main chunk"
end end
@ -72,12 +64,78 @@ debug.getinfo = function(thread, f, what)
end end
return info return info
end end
local print_error local enhance_error
print_error = function(error_message, start_fn, stop_fn) enhance_error = function(error_message, start_fn, stop_fn)
io.stderr:write(tostring(RED) .. "ERROR: " .. tostring(BRIGHT_RED) .. tostring(error_message or "") .. tostring(RESET) .. "\n") if not (error_message and error_message:match("\x1b")) then
io.stderr:write("stack traceback:\n") error_message = error_message or ""
local level = 1 do
local found_start = false 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
while true do while true do
local _continue_0 = false local _continue_0 = false
repeat repeat
@ -85,19 +143,15 @@ print_error = function(error_message, start_fn, stop_fn)
if not calling_fn then if not calling_fn then
break break
end end
level = level + 1 if calling_fn.func == xpcall then
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"
end end
if name == "run_lua_fn" then if name == "function 'run_lua_fn'" then
_continue_0 = true _continue_0 = true
break break
end end
@ -120,16 +174,7 @@ print_error = function(error_message, start_fn, stop_fn)
end end
assert(filename) assert(filename)
if calling_fn.name then if calling_fn.name then
do name = "action '" .. tostring(calling_fn.name:from_lua_id()) .. "'"
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 else
name = "main chunk" name = "main chunk"
end end
@ -192,10 +237,7 @@ print_error = function(error_message, start_fn, stop_fn)
end end
if file and (calling_fn.short_src:match("%.moon$") or file:match("^#![^\n]*moon\n")) and type(MOON_SOURCE_MAP[file]) == 'table' then if file and (calling_fn.short_src:match("%.moon$") or file:match("^#![^\n]*moon\n")) and type(MOON_SOURCE_MAP[file]) == 'table' then
local char = MOON_SOURCE_MAP[file][calling_fn.currentline] local char = MOON_SOURCE_MAP[file][calling_fn.currentline]
line_num = 1 line_num = file:line_number_at(char)
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) line = tostring(CYAN) .. tostring(calling_fn.short_src) .. ":" .. tostring(line_num) .. " in " .. tostring(name or '?') .. tostring(RESET)
else else
line_num = calling_fn.currentline line_num = calling_fn.currentline
@ -216,12 +258,9 @@ print_error = function(error_message, start_fn, stop_fn)
end end
end end
end end
io.stderr:write(line, "\n") table.insert(ret, line)
if calling_fn.istailcall then if calling_fn.istailcall then
io.stderr:write(" " .. tostring(DIM) .. "(...tail calls...)" .. tostring(RESET) .. "\n") table.insert(ret, " " .. tostring(DIM) .. "(...tail calls...)" .. tostring(RESET))
end
if calling_fn.func == stop_fn then
break
end end
_continue_0 = true _continue_0 = true
until true until true
@ -229,19 +268,20 @@ print_error = function(error_message, start_fn, stop_fn)
break break
end end
end end
return io.stderr:flush() return table.concat(ret, "\n")
end end
local guard local guard
guard = function(fn) guard = function(fn)
local error_handler local err
error_handler = function(error_message) ok, err = xpcall(fn, enhance_error)
print_error(error_message, error_handler, fn) if not ok then
local EXIT_FAILURE = 1 io.stderr:write(err)
return os.exit(EXIT_FAILURE) io.stderr:flush()
return os.exit(1)
end end
return xpcall(fn, error_handler)
end end
return { return {
guard = guard, guard = guard,
enhance_error = enhance_error,
print_error = print_error print_error = print_error
} }

View File

@ -1,6 +1,7 @@
-- This file contains the logic for making nicer error messages -- This file contains the logic for making nicer error messages
debug_getinfo = debug.getinfo debug_getinfo = debug.getinfo
Files = require "files" Files = require "files"
pretty_error = require("pretty_errors")
export SOURCE_MAP export SOURCE_MAP
RED = "\027[31m" RED = "\027[31m"
@ -31,7 +32,7 @@ debug.getinfo = (thread,f,what)->
else debug_getinfo(thread,f,what) else debug_getinfo(thread,f,what)
if not info or not info.func then return info if not info or not info.func then return info
if info.short_src or info.source or info.linedefine or info.currentline if info.short_src or info.source or info.linedefine or info.currentline
-- TODO: get name properly -- TODO: reduce duplicate code
if map = SOURCE_MAP[info.source] if map = SOURCE_MAP[info.source]
if info.currentline if info.currentline
info.currentline = assert(map[info.currentline]) info.currentline = assert(map[info.currentline])
@ -40,31 +41,71 @@ 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 info.name = if info.name
if tmp = info.name\match("^A_([a-zA-Z0-9_]*)$") "action '#{calling_fn.name\from_lua_id!}'"
tmp\gsub("_"," ")\gsub("x([0-9A-F][0-9A-F])", => string.char(tonumber(@, 16)))
else info.name
else "main chunk" else "main chunk"
return info return info
print_error = (error_message, start_fn, stop_fn)-> enhance_error = (error_message, start_fn, stop_fn)->
io.stderr\write("#{RED}ERROR: #{BRIGHT_RED}#{error_message or ""}#{RESET}\n") unless error_message and error_message\match("\x1b")
io.stderr\write("stack traceback:\n") error_message or= ""
if fn = error_message\match("attempt to call a nil value %(global '(.*)'%)")
if fn\match "x[0-9A-F][0-9A-F]"
error_message = "The action '#{fn\from_lua_id!}' is not defined."
level = 2
while true
-- TODO: reduce duplicate code
calling_fn = debug_getinfo(level)
if not calling_fn then break
level += 1
local filename, file, line_num
if map = SOURCE_MAP and SOURCE_MAP[calling_fn.source]
if calling_fn.currentline
line_num = assert(map[calling_fn.currentline])
filename,start,stop = calling_fn.source\match('@([^[]*)%[([0-9]+):([0-9]+)]')
if not filename
filename,start = calling_fn.source\match('@([^[]*)%[([0-9]+)]')
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'
char = MOON_SOURCE_MAP[file][calling_fn.currentline]
line_num = file\line_number_at(char)
else
line_num = calling_fn.currentline
level = 1 if file and filename and line_num
found_start = false start = 1
lines = file\lines!
for i=1,line_num-1 do start += #lines[i] + 1
stop = start + #lines[line_num]
start += #lines[line_num]\match("^ *")
error_message = pretty_error{
title:"Error"
error:error_message, source:file
start:start, stop:stop, filename:filename
}
break
if calling_fn.func == xpcall then break
ret = {
"#{RED}ERROR: #{BRIGHT_RED}#{error_message or ""}#{RESET}"
"stack traceback:"
}
level = 2
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 == xpcall 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 == "function 'run_lua_fn'" then continue
line = nil line = nil
if map = SOURCE_MAP and SOURCE_MAP[calling_fn.source] if map = SOURCE_MAP and SOURCE_MAP[calling_fn.source]
if calling_fn.currentline if calling_fn.currentline
@ -78,11 +119,8 @@ print_error = (error_message, start_fn, stop_fn)->
if not filename if not filename
filename,start = calling_fn.source\match('@([^[]*)%[([0-9]+)]') filename,start = calling_fn.source\match('@([^[]*)%[([0-9]+)]')
assert(filename) assert(filename)
-- TODO: get name properly
name = if calling_fn.name name = if calling_fn.name
if tmp = calling_fn.name\match("^A_([a-zA-Z0-9_]*)$") "action '#{calling_fn.name\from_lua_id!}'"
"action '#{tmp\gsub("_"," ")\gsub("x([0-9A-F][0-9A-F])", => string.char(tonumber(@, 16)))}'"
else "action '#{calling_fn.name}'"
else "main chunk" else "main chunk"
file = Files.read(filename) file = Files.read(filename)
@ -123,8 +161,7 @@ print_error = (error_message, start_fn, stop_fn)->
if file and (calling_fn.short_src\match("%.moon$") or file\match("^#![^\n]*moon\n")) and type(MOON_SOURCE_MAP[file]) == 'table' if file and (calling_fn.short_src\match("%.moon$") or file\match("^#![^\n]*moon\n")) and type(MOON_SOURCE_MAP[file]) == 'table'
char = MOON_SOURCE_MAP[file][calling_fn.currentline] char = MOON_SOURCE_MAP[file][calling_fn.currentline]
line_num = 1 line_num = file\line_number_at(char)
for _ in file\sub(1,char)\gmatch("\n") do line_num += 1
line = "#{CYAN}#{calling_fn.short_src}:#{line_num} in #{name or '?'}#{RESET}" line = "#{CYAN}#{calling_fn.short_src}:#{line_num} in #{name or '?'}#{RESET}"
else else
line_num = calling_fn.currentline line_num = calling_fn.currentline
@ -137,18 +174,17 @@ print_error = (error_message, start_fn, stop_fn)->
if err_line = lines[line_num] if err_line = lines[line_num]
offending_statement = "#{BRIGHT_RED}#{err_line\match("^[ ]*(.*)$")}#{RESET}" offending_statement = "#{BRIGHT_RED}#{err_line\match("^[ ]*(.*)$")}#{RESET}"
line ..= "\n "..offending_statement line ..= "\n "..offending_statement
io.stderr\write(line,"\n") table.insert ret, line
if calling_fn.istailcall if calling_fn.istailcall
io.stderr\write(" #{DIM}(...tail calls...)#{RESET}\n") table.insert ret, " #{DIM}(...tail calls...)#{RESET}"
if calling_fn.func == stop_fn then break
io.stderr\flush! return table.concat(ret, "\n")
guard = (fn)-> guard = (fn)->
error_handler = (error_message)-> ok, err = xpcall(fn, enhance_error)
print_error error_message, error_handler, fn if not ok
EXIT_FAILURE = 1 io.stderr\write err
os.exit(EXIT_FAILURE) io.stderr\flush!
xpcall(fn, error_handler) os.exit 1
return {:guard, :print_error} return {:guard, :enhance_error, :print_error}

View File

@ -25,7 +25,7 @@ use "core/control_flow"
local _a, _b = \($a as lua expr), \($b as lua expr) local _a, _b = \($a as lua expr), \($b as lua expr)
if _a ~= _b then if _a ~= _b then
at_1_fail(\(quote "\($a.source)"), at_1_fail(\(quote "\($a.source)"),
"Assumption failed: This value was "..tostring(_a).." when it was supposed to be "..tostring(_b)..".") "Assumption failed: This value was "..tostring(_a).." but it was expected to be "..tostring(_b)..".")
end end
end end
") ")
@ -35,7 +35,7 @@ use "core/control_flow"
local _a, _b = \($a as lua expr), \($b as lua expr) local _a, _b = \($a as lua expr), \($b as lua expr)
if _a == _b then if _a == _b then
at_1_fail(\(quote "\($a.source)"), at_1_fail(\(quote "\($a.source)"),
"Assumption failed: This value was "..tostring(_a).." when it wasn't supposed to be.") "Assumption failed: This value was "..tostring(_a).." but it wasn't expected to be.")
end end
end end
") ")
@ -76,10 +76,10 @@ test:
Lua (" Lua ("
do do
local _fell_through = false local _fell_through = false
local _result = {pcall(function() local _result = {xpcall(function()
\($action as lua) \($action as lua)
_fell_through = true _fell_through = true
end)} end, enhance_error)}
if _result[1] then if _result[1] then
\$success_lua \$success_lua
else else
@ -123,10 +123,10 @@ test:
(do $action then always $final_action) compiles to (" (do $action then always $final_action) compiles to ("
do -- do/then always do -- do/then always
local _fell_through = false local _fell_through = false
local _results = {pcall(function() local _results = {xpcall(function()
\($action as lua) \($action as lua)
_fell_through = true _fell_through = true
end)} end, enhance_error)}
\($final_action as lua) \($final_action as lua)
if not _results[1] then error(_results[2], 0) end if not _results[1] then error(_results[2], 0) end
if not _fell_through then return table.unpack(_results, 2) end if not _fell_through then return table.unpack(_results, 2) end

View File

@ -10,6 +10,7 @@ do
end end
local SyntaxTree = require("syntax_tree") local SyntaxTree = require("syntax_tree")
local Files = require("files") local Files = require("files")
local Errhand = require("error_handling")
local make_parser = require("parser") local make_parser = require("parser")
local pretty_error = require("pretty_errors") local pretty_error = require("pretty_errors")
local make_tree local make_tree
@ -137,6 +138,7 @@ nomsu_environment = Importer({
NomsuCode_from = (function(src, ...) NomsuCode_from = (function(src, ...)
return NomsuCode:from(src, ...) return NomsuCode:from(src, ...)
end), end),
enhance_error = Errhand.enhance_error,
SOURCE_MAP = { }, SOURCE_MAP = { },
getfenv = getfenv, getfenv = getfenv,
_1_as_nomsu = tree_to_nomsu, _1_as_nomsu = tree_to_nomsu,

View File

@ -4,6 +4,7 @@
{:List, :Dict, :Text} = require 'containers' {:List, :Dict, :Text} = require 'containers'
SyntaxTree = require "syntax_tree" SyntaxTree = require "syntax_tree"
Files = require "files" Files = require "files"
Errhand = require "error_handling"
make_parser = require("parser") make_parser = require("parser")
pretty_error = require("pretty_errors") pretty_error = require("pretty_errors")
@ -61,6 +62,7 @@ nomsu_environment = Importer{
:LuaCode, :NomsuCode, :Source :LuaCode, :NomsuCode, :Source
LuaCode_from: ((src, ...)-> LuaCode\from(src, ...)), LuaCode_from: ((src, ...)-> LuaCode\from(src, ...)),
NomsuCode_from: ((src, ...)-> NomsuCode\from(src, ...)), NomsuCode_from: ((src, ...)-> NomsuCode\from(src, ...)),
enhance_error: Errhand.enhance_error
SOURCE_MAP: {}, SOURCE_MAP: {},
getfenv:getfenv, getfenv:getfenv,