2018-06-19 01:12:43 -07:00
|
|
|
-- This file contains the logic for making nicer error messages
|
|
|
|
debug_getinfo = debug.getinfo
|
2019-01-14 15:42:48 -08:00
|
|
|
Files = require "files"
|
2019-01-29 16:16:45 -08:00
|
|
|
C = require "colors"
|
2019-01-18 20:46:04 -08:00
|
|
|
pretty_error = require("pretty_errors")
|
2018-06-19 01:12:43 -07:00
|
|
|
export SOURCE_MAP
|
|
|
|
|
|
|
|
ok, to_lua = pcall -> require('moonscript.base').to_lua
|
2018-06-19 02:00:52 -07:00
|
|
|
if not ok then to_lua = -> nil
|
|
|
|
MOON_SOURCE_MAP = setmetatable {},
|
|
|
|
__index: (file)=>
|
|
|
|
_, line_table = to_lua(file)
|
|
|
|
self[file] = line_table or false
|
|
|
|
return line_table or false
|
2018-06-19 01:12:43 -07:00
|
|
|
|
|
|
|
-- Make a better version of debug.getinfo that provides info about the original source
|
|
|
|
-- where the error came from, even if that's in another language.
|
|
|
|
debug.getinfo = (thread,f,what)->
|
|
|
|
if what == nil
|
|
|
|
f,what,thread = thread,f,nil
|
|
|
|
if type(f) == 'number' then f += 1 -- Account for this wrapper function
|
|
|
|
info = if thread == nil
|
|
|
|
debug_getinfo(f,what)
|
|
|
|
else debug_getinfo(thread,f,what)
|
|
|
|
if not info or not info.func then return info
|
|
|
|
if info.short_src or info.source or info.linedefine or info.currentline
|
2019-01-18 20:46:04 -08:00
|
|
|
-- TODO: reduce duplicate code
|
2018-06-19 01:12:43 -07:00
|
|
|
if map = SOURCE_MAP[info.source]
|
|
|
|
if info.currentline
|
|
|
|
info.currentline = assert(map[info.currentline])
|
|
|
|
if info.linedefined
|
|
|
|
info.linedefined = assert(map[info.linedefined])
|
|
|
|
if info.lastlinedefined
|
|
|
|
info.lastlinedefined = assert(map[info.lastlinedefined])
|
|
|
|
info.short_src = info.source\match('@([^[]*)') or info.short_src
|
2018-07-10 15:00:01 -07:00
|
|
|
info.name = if info.name
|
2019-01-19 18:27:30 -08:00
|
|
|
"action '#{info.name\from_lua_id!}'"
|
2018-07-10 15:00:01 -07:00
|
|
|
else "main chunk"
|
2018-06-19 01:12:43 -07:00
|
|
|
return info
|
|
|
|
|
2019-02-02 19:42:14 -08:00
|
|
|
-- This uses a slightly modified Damerau-Levenshtein distance:
|
|
|
|
strdist = (a,b,cache={})->
|
|
|
|
if a == b then return 0
|
|
|
|
if #a < #b then a,b = b,a
|
|
|
|
if b == "" then return #a
|
|
|
|
k = a..'\003'..b
|
|
|
|
unless cache[k]
|
|
|
|
-- Insert, delete, substitute (given weight 1.1 as a heuristic)
|
|
|
|
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)
|
|
|
|
)
|
|
|
|
-- Transposition:
|
|
|
|
if #a >= 2 and #b >= 2 and a\sub(-1,-1) == b\sub(-2,-2) and a\sub(-2,-2) == b\sub(-1,-1)
|
|
|
|
cache[k] = math.min(cache[k], strdist(a\sub(1,-3),b\sub(1,-3),cache) + 1)
|
|
|
|
return cache[k]
|
|
|
|
|
|
|
|
enhance_error = (error_message)->
|
2019-01-29 16:16:45 -08:00
|
|
|
-- Hacky: detect the line numbering
|
|
|
|
unless error_message and error_message\match("%d|")
|
2019-01-18 20:46:04 -08:00
|
|
|
error_message or= ""
|
2019-02-02 19:43:13 -08:00
|
|
|
-- When calling 'nil' actions, make a better error message
|
2019-03-04 14:19:44 -08:00
|
|
|
if fn_name = error_message\match("attempt to call a nil value %(method '(.*)'%)")
|
|
|
|
action_name = fn_name\from_lua_id!
|
|
|
|
error_message = "This object does not have the method '#{action_name}'."
|
|
|
|
elseif fn_name = (error_message\match("attempt to call a nil value %(global '(.*)'%)") or
|
2019-01-29 16:16:45 -08:00
|
|
|
error_message\match("attempt to call global '(.*)' %(a nil value%)"))
|
2019-02-02 19:42:14 -08:00
|
|
|
|
|
|
|
action_name = fn_name\from_lua_id!
|
|
|
|
error_message = "The action '#{action_name}' is not defined."
|
|
|
|
|
2019-02-02 19:43:13 -08:00
|
|
|
-- Look for simple misspellings:
|
|
|
|
|
2019-02-02 19:42:14 -08:00
|
|
|
-- This check is necessary for handling both top-level code and code inside a fn
|
|
|
|
func = debug.getinfo(2,'f').func
|
2019-03-14 17:57:30 -07:00
|
|
|
local env
|
|
|
|
if _VERSION == "Lua 5.1"
|
|
|
|
env = getfenv(func)
|
|
|
|
else
|
2019-02-02 19:42:14 -08:00
|
|
|
ename,env = debug.getupvalue(func, 1)
|
2019-03-14 17:57:30 -07:00
|
|
|
unless ename == "_ENV" or ename == "_G"
|
|
|
|
func = debug.getinfo(3,'f').func
|
|
|
|
ename,env = debug.getupvalue(func, 1)
|
2019-02-02 19:42:14 -08:00
|
|
|
|
|
|
|
THRESHOLD = math.min(4.5, .9*#action_name) -- Ignore matches with strdist > THRESHOLD
|
|
|
|
candidates = {}
|
|
|
|
cache = {}
|
|
|
|
|
|
|
|
-- Locals:
|
|
|
|
for i=1,99
|
|
|
|
k, v = debug.getlocal(2, i)
|
|
|
|
break if k == nil
|
|
|
|
unless k\sub(1,1) == "(" or type(v) != 'function'
|
|
|
|
k = k\from_lua_id!
|
|
|
|
if strdist(k, action_name, cache) <= THRESHOLD and k != ""
|
|
|
|
table.insert candidates, k
|
|
|
|
|
|
|
|
-- Upvalues:
|
|
|
|
for i=1,debug.getinfo(func, 'u').nups
|
|
|
|
k, v = debug.getupvalue(func, i)
|
|
|
|
unless k\sub(1,1) == "(" or type(v) != 'function'
|
|
|
|
k = k\from_lua_id!
|
|
|
|
if strdist(k, action_name, cache) <= THRESHOLD and k != ""
|
|
|
|
table.insert candidates, k
|
|
|
|
|
|
|
|
-- Globals and global compile rules:
|
|
|
|
scan = (t, is_lua_id)->
|
2019-03-27 14:37:38 -07:00
|
|
|
return unless t
|
2019-02-02 19:42:14 -08:00
|
|
|
for k,v in pairs(t)
|
|
|
|
if type(k) == 'string' and type(v) == 'function'
|
|
|
|
k = k\from_lua_id! unless is_lua_id
|
|
|
|
if strdist(k, action_name, cache) <= THRESHOLD and k != ""
|
|
|
|
table.insert candidates, k
|
|
|
|
scan env.COMPILE_RULES, true
|
|
|
|
scan env.COMPILE_RULES._IMPORTS, true
|
|
|
|
scan env
|
|
|
|
scan env._IMPORTS
|
|
|
|
|
|
|
|
if #candidates > 0
|
|
|
|
for c in *candidates do THRESHOLD = math.min(THRESHOLD, strdist(c, action_name, cache))
|
|
|
|
candidates = [c for c in *candidates when strdist(c, action_name, cache) <= THRESHOLD]
|
|
|
|
--candidates = ["#{c}[#{strdist(c,action_name,cache)}/#{THRESHOLD}]" for c in *candidates]
|
|
|
|
if #candidates == 1
|
|
|
|
error_message ..= "\n\x1b[3mSuggestion: Maybe you meant '#{candidates[1]}'? "
|
|
|
|
elseif #candidates > 0
|
|
|
|
last = table.remove(candidates)
|
|
|
|
error_message ..= "\n"..C('italic', "Suggestion: Maybe you meant '#{table.concat candidates, "', '"}'#{#candidates > 1 and ',' or ''} or '#{last}'? ")
|
|
|
|
|
2019-01-18 20:46:04 -08:00
|
|
|
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
|
|
|
|
|
|
|
|
if file and filename and line_num
|
|
|
|
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 = {
|
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:"
|
|
|
|
}
|
2018-06-19 01:12:43 -07:00
|
|
|
|
2019-01-18 20:46:04 -08:00
|
|
|
level = 2
|
2018-06-19 01:12:43 -07:00
|
|
|
while true
|
|
|
|
-- TODO: reduce duplicate code
|
|
|
|
calling_fn = debug_getinfo(level)
|
|
|
|
if not calling_fn then break
|
2019-01-18 20:46:04 -08:00
|
|
|
if calling_fn.func == xpcall then break
|
2018-06-19 01:12:43 -07:00
|
|
|
level += 1
|
|
|
|
name = calling_fn.name and "function '#{calling_fn.name}'" or nil
|
|
|
|
if calling_fn.linedefined == 0 then name = "main chunk"
|
2019-01-18 20:46:04 -08:00
|
|
|
if name == "function 'run_lua_fn'" then continue
|
2018-06-19 01:12:43 -07:00
|
|
|
line = nil
|
2019-01-08 16:35:32 -08:00
|
|
|
if map = SOURCE_MAP and SOURCE_MAP[calling_fn.source]
|
2018-06-19 01:12:43 -07:00
|
|
|
if calling_fn.currentline
|
|
|
|
calling_fn.currentline = assert(map[calling_fn.currentline])
|
|
|
|
if calling_fn.linedefined
|
|
|
|
calling_fn.linedefined = assert(map[calling_fn.linedefined])
|
|
|
|
if calling_fn.lastlinedefined
|
|
|
|
calling_fn.lastlinedefined = assert(map[calling_fn.lastlinedefined])
|
|
|
|
--calling_fn.short_src = calling_fn.source\match('"([^[]*)')
|
|
|
|
filename,start,stop = calling_fn.source\match('@([^[]*)%[([0-9]+):([0-9]+)]')
|
2018-12-18 17:35:27 -08:00
|
|
|
if not filename
|
|
|
|
filename,start = calling_fn.source\match('@([^[]*)%[([0-9]+)]')
|
2018-06-19 01:12:43 -07:00
|
|
|
assert(filename)
|
2018-06-19 02:00:52 -07:00
|
|
|
name = if calling_fn.name
|
2019-01-18 20:46:04 -08:00
|
|
|
"action '#{calling_fn.name\from_lua_id!}'"
|
2018-06-19 02:00:52 -07:00
|
|
|
else "main chunk"
|
2018-07-23 15:25:23 -07:00
|
|
|
|
2019-01-14 15:42:48 -08:00
|
|
|
file = Files.read(filename)
|
|
|
|
lines = file and file\lines! or {}
|
|
|
|
if err_line = lines[calling_fn.currentline]
|
2019-01-29 16:16:45 -08:00
|
|
|
offending_statement = C('bright red', err_line\match("^[ ]*(.*)"))
|
|
|
|
line = C('yellow', "#{filename}:#{calling_fn.currentline} in #{name}\n #{offending_statement}")
|
2018-07-23 15:25:23 -07:00
|
|
|
else
|
2019-01-29 16:16:45 -08:00
|
|
|
line = C('yellow', "#{filename}:#{calling_fn.currentline} in #{name}")
|
2018-06-19 01:12:43 -07:00
|
|
|
else
|
|
|
|
local line_num
|
|
|
|
if name == nil
|
|
|
|
search_level = level
|
|
|
|
_info = debug.getinfo(search_level)
|
2018-07-10 15:00:01 -07:00
|
|
|
while true
|
2018-06-19 01:12:43 -07:00
|
|
|
search_level += 1
|
|
|
|
_info = debug.getinfo(search_level)
|
2018-07-10 15:00:01 -07:00
|
|
|
break unless _info
|
2018-06-19 01:12:43 -07:00
|
|
|
for i=1,999
|
|
|
|
varname, val = debug.getlocal(search_level, i)
|
|
|
|
if not varname then break
|
|
|
|
if val == calling_fn.func
|
|
|
|
name = "local '#{varname}'"
|
|
|
|
if not varname\match("%(")
|
|
|
|
break
|
|
|
|
unless name
|
|
|
|
for i=1,_info.nups
|
|
|
|
varname, val = debug.getupvalue(_info.func, i)
|
|
|
|
if not varname then break
|
|
|
|
if val == calling_fn.func
|
|
|
|
name = "upvalue '#{varname}'"
|
|
|
|
if not varname\match("%(")
|
|
|
|
break
|
2019-01-14 15:42:48 -08:00
|
|
|
|
|
|
|
local file, lines
|
|
|
|
if file = Files.read(calling_fn.short_src)
|
|
|
|
lines = file\lines!
|
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'
|
2018-06-19 02:00:52 -07:00
|
|
|
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', "#{calling_fn.short_src}:#{line_num} in #{name or '?'}")
|
2018-06-19 01:12:43 -07:00
|
|
|
else
|
|
|
|
line_num = calling_fn.currentline
|
|
|
|
if calling_fn.short_src == '[C]'
|
2019-01-29 16:16:45 -08:00
|
|
|
line = C('green', "#{calling_fn.short_src} in #{name or '?'}")
|
2018-06-19 01:12:43 -07:00
|
|
|
else
|
2019-01-29 16:16:45 -08:00
|
|
|
line = C('blue', "#{calling_fn.short_src}:#{calling_fn.currentline} in #{name or '?'}")
|
2018-06-19 01:12:43 -07:00
|
|
|
|
|
|
|
if file
|
2019-01-14 15:42:48 -08:00
|
|
|
if err_line = lines[line_num]
|
2019-01-29 16:16:45 -08:00
|
|
|
offending_statement = C('bright red', "#{err_line\match("^[ ]*(.*)$")}")
|
2018-07-23 15:25:23 -07:00
|
|
|
line ..= "\n "..offending_statement
|
2019-01-18 20:46:04 -08:00
|
|
|
table.insert ret, line
|
2018-06-19 01:12:43 -07:00
|
|
|
if calling_fn.istailcall
|
2019-01-29 16:16:45 -08:00
|
|
|
table.insert ret, C('dim', " (...tail calls...)")
|
2018-06-19 01:12:43 -07:00
|
|
|
|
2019-01-18 20:46:04 -08:00
|
|
|
return table.concat(ret, "\n")
|
2018-06-19 01:12:43 -07:00
|
|
|
|
2018-07-10 15:00:01 -07:00
|
|
|
guard = (fn)->
|
2019-01-18 20:46:04 -08:00
|
|
|
ok, err = xpcall(fn, enhance_error)
|
|
|
|
if not ok
|
|
|
|
io.stderr\write err
|
|
|
|
io.stderr\flush!
|
|
|
|
os.exit 1
|
2018-06-19 01:12:43 -07:00
|
|
|
|
2019-01-18 20:46:04 -08:00
|
|
|
return {:guard, :enhance_error, :print_error}
|