2018-03-13 02:02:51 -07:00
|
|
|
C = require "curses"
|
|
|
|
repr = require 'repr'
|
2018-03-13 02:55:03 -07:00
|
|
|
local REGULAR, INVERTED, HIGHLIGHTED, RED, SCREEN_H, SCREEN_W
|
|
|
|
local run_debugger, cursed, stdscr, main_loop
|
2018-03-13 02:02:51 -07:00
|
|
|
AUTO = -1
|
2018-03-13 02:55:03 -07:00
|
|
|
log = io.open("output.log", "w")
|
2018-03-13 02:02:51 -07:00
|
|
|
|
|
|
|
-- Return the callstack index of the code that actually caused an error and the max index
|
|
|
|
callstack_range = ->
|
|
|
|
min, max = 0, -1
|
|
|
|
for i=1,999 do
|
2018-03-13 02:55:03 -07:00
|
|
|
info = debug.getinfo(i, 'f')
|
|
|
|
if not info
|
|
|
|
error("Could not find info at level #{i}")
|
|
|
|
continue unless info
|
|
|
|
if info.func == main_loop
|
2018-03-13 02:02:51 -07:00
|
|
|
min = i+2
|
|
|
|
break
|
|
|
|
for i=min,999
|
2018-03-13 02:55:03 -07:00
|
|
|
info = debug.getinfo(i, 'f')
|
|
|
|
continue unless info
|
|
|
|
if info.func == cursed then
|
2018-03-13 02:02:51 -07:00
|
|
|
max = i-3
|
|
|
|
break
|
|
|
|
return min, max
|
|
|
|
|
2018-03-13 02:55:03 -07:00
|
|
|
alternating_colors = setmetatable({}, {__index:(i)=> if i % 2 == 0 then INVERTED else REGULAR})
|
2018-03-13 02:02:51 -07:00
|
|
|
class Pad
|
2018-03-13 02:55:03 -07:00
|
|
|
new: (@y,@x,@height,@width,@lines,@attrs=alternating_colors)=>
|
|
|
|
--log\write("New Pad:\n #{table.concat @lines, "\n "}\n")
|
2018-03-13 02:02:51 -07:00
|
|
|
@offset = 0
|
|
|
|
@selected = nil
|
2018-03-13 02:55:03 -07:00
|
|
|
|
|
|
|
@_height = #@lines + 2
|
|
|
|
@_width = 2
|
|
|
|
for x in *@lines do @_width = math.max(@_width, #x+2)
|
|
|
|
|
2018-03-13 02:02:51 -07:00
|
|
|
if @height == AUTO
|
2018-03-13 02:55:03 -07:00
|
|
|
@height = @_height
|
2018-03-13 02:02:51 -07:00
|
|
|
if @width == AUTO
|
2018-03-13 02:55:03 -07:00
|
|
|
@width = @_width
|
|
|
|
|
|
|
|
--log\write("#lines = #{#@lines}, height = #{@_height}, width = #{@_width}\n")
|
2018-03-13 02:02:51 -07:00
|
|
|
@_pad = C.newpad(@_height, @_width)
|
|
|
|
@_pad\scrollok(true)
|
2018-03-13 02:55:03 -07:00
|
|
|
|
|
|
|
@chstrs = {}
|
|
|
|
for i, line in ipairs(@lines)
|
|
|
|
attr = @attrs[i]
|
|
|
|
chstr = C.new_chstr(@width-2)
|
|
|
|
@chstrs[i] = chstr
|
|
|
|
chstr\set_str(0, line, attr)
|
|
|
|
chstr\set_str(#line+0, ' ', attr, chstr\len!-#line)
|
|
|
|
@_pad\mvaddchstr(i-1+1,0+1,chstr)
|
2018-03-13 02:02:51 -07:00
|
|
|
@refresh!
|
|
|
|
|
|
|
|
select: (i)=>
|
2018-03-13 02:55:03 -07:00
|
|
|
if i == @selected or #@lines == 0 then return
|
2018-03-13 02:02:51 -07:00
|
|
|
if i != nil
|
|
|
|
i = math.max(1, math.min(#@lines, i))
|
|
|
|
if @selected
|
|
|
|
j = @selected
|
2018-03-13 02:55:03 -07:00
|
|
|
@chstrs[j]\set_str(0, @lines[j], @attrs[j])
|
|
|
|
@chstrs[j]\set_str(#@lines[j], ' ', @attrs[j], @chstrs[j]\len!-#@lines[j])
|
2018-03-13 02:02:51 -07:00
|
|
|
@_pad\mvaddchstr(j-1+1,0+1,@chstrs[j])
|
|
|
|
|
|
|
|
if i
|
2018-03-13 02:55:03 -07:00
|
|
|
assert(@chstrs[i], "DIDN'T FIND CHSTR: #{i}/#{#@chstrs} (#{#@lines})")
|
2018-03-13 02:02:51 -07:00
|
|
|
@chstrs[i]\set_str(0, @lines[i], HIGHLIGHTED)
|
|
|
|
@chstrs[i]\set_str(#@lines[i], ' ', HIGHLIGHTED, @chstrs[i]\len!-#@lines[i])
|
|
|
|
@_pad\mvaddchstr(i-1+1,0+1,@chstrs[i])
|
|
|
|
|
|
|
|
@selected = i
|
|
|
|
|
|
|
|
if @selected
|
|
|
|
if @offset + @height-1 < @selected
|
|
|
|
@offset = math.min(@selected - (@height-1), #@lines-@height)
|
|
|
|
elseif @offset + 1 > @selected
|
|
|
|
@offset = @selected - 1
|
|
|
|
@refresh!
|
|
|
|
if @on_select then @on_select(@selected)
|
|
|
|
|
|
|
|
refresh: =>
|
|
|
|
@_pad\border(C.ACS_VLINE, C.ACS_VLINE,
|
|
|
|
C.ACS_HLINE, C.ACS_HLINE,
|
|
|
|
C.ACS_ULCORNER, C.ACS_URCORNER,
|
|
|
|
C.ACS_LLCORNER, C.ACS_LRCORNER)
|
2018-03-13 02:55:03 -07:00
|
|
|
@_pad\pnoutrefresh(@offset,0,@y,@x,@y+@height+1,@x+@width)
|
2018-03-13 02:02:51 -07:00
|
|
|
|
|
|
|
erase: =>
|
|
|
|
@_pad\erase!
|
|
|
|
@_pad\pnoutrefresh(@offset,0,@y,@x,@y+@height,@x+@width)
|
|
|
|
|
|
|
|
clear: =>
|
|
|
|
@erase!
|
|
|
|
@lines = {}
|
|
|
|
@chstrs = {}
|
|
|
|
@set_internal_size(2,2)
|
|
|
|
if @resize_height
|
|
|
|
@set_size(@_height, @width)
|
|
|
|
if @resize_width
|
|
|
|
@set_size(@height, @_width)
|
|
|
|
@selected = nil
|
|
|
|
@offset = 0
|
|
|
|
@refresh!
|
|
|
|
|
|
|
|
scroll: (delta)=>
|
|
|
|
@select(@selected and (@selected + delta) or 1)
|
|
|
|
|
|
|
|
|
2018-03-13 02:55:03 -07:00
|
|
|
main_loop = (err_msg, stack_index=1, var_index)->
|
2018-03-13 02:02:51 -07:00
|
|
|
SCREEN_H, SCREEN_W = stdscr\getmaxyx!
|
|
|
|
|
|
|
|
stdscr\clear!
|
|
|
|
stdscr\refresh!
|
|
|
|
|
|
|
|
stack_names = {}
|
|
|
|
stack_locations = {}
|
|
|
|
max_filename = 0
|
|
|
|
stack_min, stack_max = callstack_range!
|
|
|
|
for i=stack_min,stack_max
|
|
|
|
info = debug.getinfo(i)
|
|
|
|
if not info then break
|
2018-03-13 02:55:03 -07:00
|
|
|
table.insert(stack_names, info.name or "<unnamed function>")
|
2018-03-13 02:02:51 -07:00
|
|
|
filename = info.short_src..":"..info.currentline
|
|
|
|
table.insert(stack_locations, filename)
|
|
|
|
max_filename = math.max(max_filename, #filename)
|
|
|
|
callstack = {}
|
|
|
|
for i=1,#stack_names do
|
|
|
|
callstack[i] = stack_locations[i]..(" ")\rep(max_filename-#stack_locations[i]).." | "..stack_names[i].." "
|
|
|
|
|
|
|
|
err_msg_lines = {}
|
|
|
|
for line in err_msg\gmatch("[^\n]*")
|
|
|
|
buff = ""
|
|
|
|
for word in line\gmatch("%S%S*%s*")
|
|
|
|
if #buff + #word > SCREEN_W - 4
|
|
|
|
table.insert(err_msg_lines, " "..buff)
|
|
|
|
buff = word
|
|
|
|
else
|
|
|
|
buff = buff .. word
|
|
|
|
table.insert(err_msg_lines, " "..buff)
|
2018-03-13 02:55:03 -07:00
|
|
|
err_pad = Pad(0,0,AUTO,SCREEN_W, err_msg_lines,setmetatable({}, __index:->RED))
|
|
|
|
err_pad._pad\attrset(RED)
|
|
|
|
|
|
|
|
stack_pad = Pad(err_pad.height,0,AUTO,AUTO, callstack)
|
|
|
|
stack_pad\select(stack_index)
|
|
|
|
|
|
|
|
callstack_min, _ = callstack_range!
|
|
|
|
_var_names, _var_values = {}, {}
|
|
|
|
for loc=1,999
|
|
|
|
name, value = debug.getlocal(callstack_min+stack_index-1, loc)
|
|
|
|
if value == nil then break
|
|
|
|
table.insert(_var_names, tostring(name))
|
|
|
|
if type(value) == 'function'
|
|
|
|
info = debug.getinfo(value, 'nS')
|
|
|
|
--var_values\add_line(("function: %s @ %s:%s")\format(info.name or '???', info.short_src, info.linedefined))
|
|
|
|
table.insert(_var_values, repr(info))
|
|
|
|
else
|
|
|
|
table.insert(_var_values, repr(value))
|
2018-03-13 02:02:51 -07:00
|
|
|
|
2018-03-13 02:55:03 -07:00
|
|
|
var_names = Pad(err_pad.height,stack_pad.x+stack_pad.width,10,AUTO,_var_names)
|
|
|
|
var_values = Pad(err_pad.height,var_names.x+var_names.width,10,AUTO,_var_values)
|
|
|
|
if var_index and #_var_names > 0
|
|
|
|
var_names\select(var_index)
|
|
|
|
var_values\select(var_index)
|
2018-03-13 02:02:51 -07:00
|
|
|
|
|
|
|
while true
|
|
|
|
C.doupdate!
|
|
|
|
c = stdscr\getch!
|
2018-03-13 02:55:03 -07:00
|
|
|
switch c
|
|
|
|
when C.KEY_DOWN, C.KEY_SF, ("j")\byte!
|
|
|
|
if var_index
|
|
|
|
var_names\scroll(1)
|
|
|
|
return main_loop(err_msg,stack_pad.selected,var_names.selected)
|
|
|
|
else
|
|
|
|
stack_pad\scroll(1)
|
|
|
|
return main_loop(err_msg,stack_pad.selected,nil)
|
|
|
|
|
|
|
|
when C.KEY_UP, C.KEY_SR, ("k")\byte!
|
|
|
|
if var_index
|
|
|
|
var_names\scroll(-1)
|
|
|
|
return main_loop(err_msg,stack_pad.selected,var_names.selected)
|
|
|
|
else
|
|
|
|
stack_pad\scroll(-1)
|
|
|
|
return main_loop(err_msg,stack_pad.selected,nil)
|
|
|
|
|
|
|
|
when ('J')\byte!
|
|
|
|
if var_index
|
|
|
|
var_names\scroll(10)
|
|
|
|
return main_loop(err_msg,stack_pad.selected,var_names.selected)
|
|
|
|
else
|
|
|
|
stack_pad\scroll(10)
|
|
|
|
return main_loop(err_msg,stack_pad.selected,nil)
|
|
|
|
|
|
|
|
when ('K')\byte!
|
|
|
|
if var_index
|
|
|
|
var_names\scroll(-10)
|
|
|
|
return main_loop(err_msg,stack_pad.selected,var_names.selected)
|
|
|
|
else
|
|
|
|
stack_pad\scroll(-10)
|
|
|
|
return main_loop(err_msg,stack_pad.selected,nil)
|
|
|
|
|
|
|
|
when C.KEY_RIGHT, ("l")\byte!
|
|
|
|
if var_index == nil
|
|
|
|
return main_loop(err_msg,stack_pad.selected,1)
|
|
|
|
|
|
|
|
when C.KEY_LEFT, ("h")\byte!
|
|
|
|
if var_index != nil
|
|
|
|
return main_loop(err_msg,stack_pad.selected,nil)
|
|
|
|
|
|
|
|
when ('o')\byte!
|
|
|
|
file = stack_locations[stack_pad.selected]
|
|
|
|
filename,line_no = file\match("([^:]*):(.*)")
|
|
|
|
-- Launch system editor and then redraw everything
|
|
|
|
C.endwin!
|
|
|
|
os.execute((os.getenv("EDITOR") or "nano").." +"..line_no.." "..filename)
|
|
|
|
initial_index = stack_pad.selected
|
|
|
|
return main_loop(err_msg,stack_pad.selected,var_index)
|
|
|
|
|
|
|
|
when ('q')\byte!, ("Q")\byte!
|
|
|
|
break
|
|
|
|
|
|
|
|
run_debugger = (err_msg)->
|
|
|
|
export stdscr, SCREEN_H, SCREEN_W
|
|
|
|
stdscr = C.initscr!
|
|
|
|
SCREEN_H, SCREEN_W = stdscr\getmaxyx!
|
|
|
|
|
|
|
|
C.cbreak!
|
|
|
|
C.echo(false)
|
|
|
|
C.nl(false)
|
|
|
|
C.curs_set(0)
|
|
|
|
C.start_color!
|
|
|
|
C.use_default_colors!
|
|
|
|
|
|
|
|
export REGULAR, INVERTED, HIGHLIGHTED, RED
|
|
|
|
_, REGULAR = C.init_pair(1, -1, -1), C.color_pair(1)
|
|
|
|
_, INVERTED = C.init_pair(2, -1, C.COLOR_BLACK), C.color_pair(2)
|
|
|
|
_, HIGHLIGHTED = C.init_pair(3, C.COLOR_BLACK, C.COLOR_YELLOW), C.color_pair(3)
|
|
|
|
_, RED = C.init_pair(4, C.COLOR_RED, -1), C.color_pair(4) | C.A_BOLD
|
|
|
|
|
|
|
|
return main_loop(err_msg)
|
2018-03-13 02:02:51 -07:00
|
|
|
|
|
|
|
|
|
|
|
cursed = (fn, ...)->
|
|
|
|
-- To display Lua errors, we must close curses to return to
|
|
|
|
-- normal terminal mode, and then write the error to stdout.
|
|
|
|
err_hand = (err)->
|
|
|
|
C.endwin!
|
|
|
|
print "Caught an error:"
|
|
|
|
print(debug.traceback(err, 2))
|
|
|
|
os.exit(2)
|
|
|
|
|
|
|
|
return xpcall(fn, ((err_msg)-> xpcall(run_debugger, err_hand, err_msg)), ...)
|
|
|
|
|
|
|
|
return cursed
|