lua-debug-tui/ldt.moon

544 lines
21 KiB
Plaintext
Raw Normal View History

2018-03-13 02:02:51 -07:00
C = require "curses"
re = require 're'
2018-03-13 02:02:51 -07:00
repr = require 'repr'
2018-03-21 15:03:01 -07:00
local ldb, stdscr
2018-03-14 18:45:32 -07:00
AUTO = {}
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
min = i-1
break
2018-03-21 15:03:01 -07:00
if info.func == ldb.run_debugger
2018-03-21 14:52:55 -07:00
min = i+2
2018-03-13 02:02:51 -07:00
break
for i=min,999
2018-03-13 02:55:03 -07:00
info = debug.getinfo(i, 'f')
2018-03-21 15:03:01 -07:00
if not info or info.func == ldb.guard
2018-03-21 14:52:55 -07:00
max = i-3
2018-03-13 02:02:51 -07:00
break
return min, max
2018-03-13 04:18:28 -07:00
wrap_text = (text, width)->
lines = {}
for line in text\gmatch("[^\n]*")
2018-03-13 04:18:28 -07:00
while #line > width
table.insert(lines, line\sub(1,width))
line = line\sub(width+1,-1)
if #line > 0
table.insert(lines, line)
return lines
2018-03-13 23:53:06 -07:00
2018-03-21 15:08:38 -07:00
local color
do
color_index = 0
existing = {}
make_color = (fg=-1, bg=-1)->
key = "#{fg},#{bg}"
unless existing[key]
color_index += 1
C.init_pair(color_index, fg, bg)
existing[key] = C.color_pair(color_index)
return existing[key]
color_lang = re.compile[[
x <- {|
{:attrs: {| {attr} (" " {attr})* |} :}
/ (
({:bg: "on " {color} :} / ({:fg: color :} (" on " {:bg: color :})?))
{:attrs: {| (" " {attr})* |} :})
|}
attr <- "blink" / "bold" / "dim" / "invis" / "normal" / "protect" / "reverse" / "standout" / "underline"
color <- "black" / "blue" / "cyan" / "green" / "magenta" / "red" / "white" / "yellow" / "default"
]]
C.COLOR_DEFAULT = -1
color = (s="default")->
t = assert(color_lang\match(s), "Invalid color: #{s}")
if t.fg then t.fg = C["COLOR_"..t.fg\upper!]
if t.bg then t.bg = C["COLOR_"..t.bg\upper!]
c = make_color(t.fg, t.bg)
for a in *t.attrs
c |= C["A_"..a\upper!]
return c
2018-03-13 02:02:51 -07:00
class Pad
2018-03-19 20:32:09 -07:00
new: (@label,@y,@x,height,width,...)=>
2018-03-14 18:45:32 -07:00
@scroll_y, @scroll_x = 1, 1
2018-03-13 02:02:51 -07:00
@selected = nil
2018-03-13 02:55:03 -07:00
2018-03-19 20:32:09 -07:00
@columns = {}
@column_widths = {}
2018-03-21 19:47:46 -07:00
@active_frame = color("yellow bold")
2018-03-19 20:32:09 -07:00
@inactive_frame = color("blue dim")
@colors = {}
for i=1,select('#',...)-1,2
col = select(i, ...)
table.insert(@columns, col)
w = 0
for chunk in *col do w = math.max(w, #chunk)
table.insert(@column_widths, w)
color_fn = select(i+1,...) or ((i)=>color())
assert(type(color_fn) == 'function', "Invalid color function type: #{type color_fn}")
table.insert(@colors, color_fn)
@configure_size height, width
2018-03-20 16:54:14 -07:00
log\write("New pad: #{@height},#{@width} #{@_height},#{@_width}\n")
2018-03-19 20:32:09 -07:00
@_frame = C.newwin(@height, @width, @y, @x)
@_frame\immedok(true)
@_pad = C.newpad(@_height, @_width)
@_pad\scrollok(true)
@set_active false
@chstrs = {}
for i=1,#@columns[1]
@chstrs[i] = C.new_chstr(@_width)
@setup_chstr(i)
@dirty = true
configure_size: (@height, @width)=>
2018-03-20 16:54:14 -07:00
@_height = math.max(#@columns[1], 1)
2018-03-14 18:45:32 -07:00
if @height == AUTO
@height = @_height + 2
2018-03-20 16:44:30 -07:00
@_width = #@columns-1
for i,col in ipairs(@columns)
col_width = 0
for chunk in *col do col_width = math.max(col_width, #chunk)
@_width += col_width
2018-03-20 16:54:14 -07:00
@_width = math.max(@_width, 1)
2018-03-13 23:53:06 -07:00
if @width == AUTO
2018-03-14 18:45:32 -07:00
@width = @_width + 2
2018-03-13 02:55:03 -07:00
2018-03-19 20:32:09 -07:00
setup_chstr: (i)=>
chstr = @chstrs[i]
x = 0
for c=1,#@columns
attr = @colors[c](@, i)
chunk = @columns[c][i]
chstr\set_str(x, chunk, attr)
x += #chunk
2018-03-20 15:57:21 -07:00
if #chunk < @column_widths[c]
chstr\set_str(x, " ", attr, @column_widths[c]-#chunk)
x += @column_widths[c]-#chunk
2018-03-19 20:32:09 -07:00
if c < #@columns
2018-03-21 14:23:52 -07:00
chstr\set_ch(x, C.ACS_VLINE, color("black bold"))
2018-03-19 20:32:09 -07:00
x += 1
@_pad\mvaddchstr(i-1,0,chstr)
@dirty = true
2018-03-13 02:02:51 -07:00
2018-03-13 23:53:06 -07:00
set_active: (active)=>
2018-03-14 16:10:17 -07:00
return if active == @active
2018-03-13 23:53:06 -07:00
@active = active
2018-03-19 20:32:09 -07:00
@_frame\attrset(active and @active_frame or @inactive_frame)
@dirty = true
2018-03-13 23:53:06 -07:00
2018-03-13 02:02:51 -07:00
select: (i)=>
2018-03-19 20:32:09 -07:00
if #@columns[1] == 0 then i = nil
2018-03-14 18:45:32 -07:00
if i == @selected then return @selected
2018-03-19 20:32:09 -07:00
old_y, old_x = @scroll_y, @scroll_x
2018-03-13 02:02:51 -07:00
if i != nil
2018-03-19 20:32:09 -07:00
i = math.max(1, math.min(#@columns[1], i))
old_selected,@selected = @selected,i
if old_selected
@setup_chstr(old_selected)
2018-03-13 02:02:51 -07:00
if @selected
2018-03-19 20:32:09 -07:00
@setup_chstr(@selected)
2018-03-13 02:02:51 -07:00
2018-03-14 18:45:32 -07:00
scrolloff = 3
2018-03-19 20:32:09 -07:00
if @selected > @scroll_y + (@height-2) - scrolloff
@scroll_y = @selected - (@height-2) + scrolloff
elseif @selected < @scroll_y + scrolloff
@scroll_y = @selected - scrolloff
2018-03-21 19:22:25 -07:00
@scroll_y = math.max(1, math.min(@_height, @scroll_y))
2018-03-19 20:32:09 -07:00
if @scroll_y == old_y
w = math.min(@width-2,@_width)
if old_selected and @scroll_y <= old_selected and old_selected <= @scroll_y + @height-2
@_pad\pnoutrefresh(old_selected-1,@scroll_x-1,@y+1+(old_selected-@scroll_y),@x+1,@y+1+(old_selected-@scroll_y)+1,@x+w)
if @selected and @scroll_y <= @selected and @selected <= @scroll_y + @height-2
@_pad\pnoutrefresh(@selected-1,@scroll_x-1,@y+1+(@selected-@scroll_y),@x+1,@y+1+(@selected-@scroll_y)+1,@x+w)
else
@dirty = true
2018-03-13 02:02:51 -07:00
if @on_select then @on_select(@selected)
return @selected
2018-03-13 02:02:51 -07:00
2018-03-14 18:45:32 -07:00
scroll: (dy,dx)=>
2018-03-19 20:32:09 -07:00
old_y, old_x = @scroll_y, @scroll_x
2018-03-14 18:45:32 -07:00
if @selected != nil
@select(@selected + (dy or 0))
else
2018-03-21 19:22:25 -07:00
@scroll_y = math.max(1, math.min(@_height-(@height-2-1), @scroll_y+(dy or 0)))
@scroll_x = math.max(1, math.min(@_width-(@width-2-1), @scroll_x+(dx or 0)))
2018-03-19 20:32:09 -07:00
if @scroll_y != old_y or @scroll_x != old_x
@dirty = true
2018-03-14 16:10:17 -07:00
2018-03-19 20:32:09 -07:00
refresh: (force=false)=>
return if not force and not @dirty
2018-03-14 18:45:32 -07:00
@_frame\border(C.ACS_VLINE, C.ACS_VLINE,
2018-03-13 02:02:51 -07:00
C.ACS_HLINE, C.ACS_HLINE,
C.ACS_ULCORNER, C.ACS_URCORNER,
C.ACS_LLCORNER, C.ACS_LRCORNER)
2018-03-13 23:53:06 -07:00
if @label
2018-03-14 18:45:32 -07:00
@_frame\mvaddstr(0, math.floor((@width-#@label-2)/2), " #{@label} ")
@_frame\refresh!
2018-03-19 20:32:09 -07:00
--@_frame\prefresh(0,0,@y,@x,@y+@height-1,@x+@width-1)
2018-03-14 18:45:32 -07:00
h,w = math.min(@height-2,@_height),math.min(@width-2,@_width)
@_pad\pnoutrefresh(@scroll_y-1,@scroll_x-1,@y+1,@x+1,@y+h,@x+w)
2018-03-19 20:32:09 -07:00
@dirty = false
2018-03-13 02:02:51 -07:00
erase: =>
2018-03-19 20:32:09 -07:00
@dirty = true
2018-03-14 18:45:32 -07:00
@_frame\erase!
@_frame\refresh!
2018-03-13 02:02:51 -07:00
2018-03-14 18:45:32 -07:00
__gc: =>
2018-03-15 15:30:43 -07:00
@_frame\close!
@_pad\close!
2018-03-13 02:02:51 -07:00
2018-03-19 20:32:09 -07:00
class NumberedPad extends Pad
new: (@label,@y,@x,height,width,...)=>
col1 = select(1, ...)
fmt = "%#{#tostring(#col1)}d"
line_nums = [fmt\format(i) for i=1,#col1]
2018-03-21 19:47:46 -07:00
cols = {line_nums, ((i)=> i == @selected and color() or color("yellow")), ...}
2018-03-19 20:32:09 -07:00
super @label, @y, @x, height, width, unpack(cols)
ok, to_lua = pcall -> require('moonscript.base').to_lua
if not ok then to_lua = -> nil
file_cache = setmetatable({}, {__index:(filename)=>
file = io.open(filename)
if not file then return nil
contents = file\read("*a")
@[filename] = contents
return contents
})
line_tables = setmetatable({}, {__index:(filename)=>
file = file_cache[filename]
2018-03-13 15:51:28 -07:00
if not file
return nil
ok, line_table = to_lua(file)
if ok
@[filename] = line_table
return line_table
})
2018-03-13 15:55:14 -07:00
2018-03-21 15:03:01 -07:00
-- Cleanup curses and print the error to stdout like regular
2018-03-21 14:52:55 -07:00
err_hand = (err)->
C.endwin!
print "Error in debugger."
print(debug.traceback(err, 2))
os.exit(2)
2018-03-21 15:03:01 -07:00
ldb = {
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!
do -- Fullscreen flash
stdscr\wbkgd(color"yellow on red bold")
stdscr\clear!
stdscr\refresh!
lines = wrap_text("ERROR!\n \n "..err_msg.."\n \npress any key...", math.floor(SCREEN_W/2))
max_line = 0
for line in *lines do max_line = math.max(max_line, #line)
for i, line in ipairs(lines)
if i == 1 or i == #lines
stdscr\mvaddstr(math.floor(SCREEN_H/2 - #lines/2)+i, math.floor((SCREEN_W-#line)/2), line)
else
stdscr\mvaddstr(math.floor(SCREEN_H/2 - #lines/2)+i, math.floor((SCREEN_W-max_line)/2), line)
stdscr\refresh!
C.doupdate!
stdscr\getch!
2018-03-21 19:30:36 -07:00
stdscr\keypad!
2018-03-21 15:03:01 -07:00
stdscr\wbkgd(color!)
stdscr\clear!
stdscr\refresh!
pads = {}
do -- Err pad
err_msg_lines = wrap_text(err_msg, SCREEN_W - 4)
for i,line in ipairs(err_msg_lines)
err_msg_lines[i] = (" ")\rep(2)..line
pads.err = Pad("Error Message", 0,0,AUTO,SCREEN_W, err_msg_lines, (i)=> color("red bold"))
pads.err._frame\attrset(color("red"))
pads.err\refresh!
stack_locations = {}
2018-03-21 19:22:25 -07:00
err_lines = {}
2018-03-21 15:03:01 -07:00
do -- Stack pad
stack_names = {}
max_filename, max_fn_name = 0, 0
stack_min, stack_max = callstack_range!
for i=stack_min,stack_max
info = debug.getinfo(i)
if not info then break
fn_name = info.name or "<unnamed function>"
table.insert(stack_names, fn_name)
line = if info.short_src
line_table = line_tables[info.short_src]
if line_table
char = line_table[info.currentline]
line_num = 1
file = file_cache[info.short_src]
for _ in file\sub(1,char)\gmatch("\n") do line_num += 1
"#{info.short_src}:#{line_num}"
else
info.short_src..":"..info.currentline
else
"???"
2018-03-21 19:22:25 -07:00
err_lines[line] = true
2018-03-21 15:03:01 -07:00
table.insert(stack_locations, line)
max_filename = math.max(max_filename, #line)
max_fn_name = math.max(max_fn_name, #fn_name)
max_fn_name, max_filename = 0, 0
for i=1,#stack_names do
2018-03-21 19:47:46 -07:00
max_fn_name = math.max(max_fn_name, #stack_names[i])
2018-03-21 15:03:01 -07:00
max_filename = math.max(max_filename, #stack_locations[i])
2018-03-21 19:47:46 -07:00
stack_h = math.max(#stack_names+2, math.floor(2/3*SCREEN_H))
stack_w = max_fn_name + 3 + max_filename
2018-03-21 15:03:01 -07:00
pads.stack = Pad "(C)allstack",pads.err.height,SCREEN_W-stack_w,stack_h,stack_w,
stack_names, ((i)=> (i == @selected) and color("black on green") or color("green bold")),
stack_locations, ((i)=> (i == @selected) and color("black on cyan") or color("cyan bold"))
show_src = (filename, line_no)->
if pads.src
2018-03-25 15:14:18 -07:00
if pads.src.filename == filename
pads.src\select(line_no)
pads.src.colors[2] = (i)=>
return if i == line_no and i == @selected then color("yellow on red bold")
elseif i == line_no then color("yellow on red")
elseif err_lines["#{filename}:#{i}"] == true then color("red on black bold")
elseif i == @selected then color("reverse")
else color()
for line,_ in pairs(err_lines)
pads.src\setup_chstr(tonumber(line\match("[^:]*:(%d*).*")))
pads.src\select(line_no)
return
else
pads.src\erase!
2018-03-21 15:03:01 -07:00
file = file_cache[filename]
if file
src_lines = {}
for line in (file..'\n')\gmatch("([^\n]*)\n")
table.insert src_lines, line
pads.src = NumberedPad "(S)ource Code", pads.err.height,0,
pads.stack.height,pads.stack.x, src_lines, (i)=>
2018-03-21 19:22:25 -07:00
return if i == line_no and i == @selected then color("yellow on red bold")
elseif i == line_no then color("yellow on red")
elseif err_lines["#{filename}:#{i}"] == true then color("red on black bold")
2018-03-21 19:47:46 -07:00
elseif i == @selected then color("reverse")
else color()
2018-03-21 15:03:01 -07:00
pads.src\select(line_no)
else
lines = {}
for i=1,math.floor(pads.stack.height/2)-1 do table.insert(lines, "")
s = "<no source code found>"
s = (" ")\rep(math.floor((pads.stack.x-2-#s)/2))..s
table.insert(lines, s)
pads.src = Pad "(S)ource Code", pads.err.height,0,pads.stack.height,pads.stack.x,lines, ->color("red")
2018-03-25 15:14:18 -07:00
pads.src.filename = filename
2018-03-21 15:03:01 -07:00
show_vars = (stack_index)->
if pads.vars
pads.vars\erase!
if pads.values
pads.values\erase!
callstack_min, _ = callstack_range!
var_names, 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))
2018-03-21 19:22:25 -07:00
table.insert(values, value)
[[
2018-03-21 15:03:01 -07:00
if type(value) == 'function'
info = debug.getinfo(value, 'nS')
--values\add_line(("function: %s @ %s:%s")\format(info.name or '???', info.short_src, info.linedefined))
table.insert(values, repr(info))
else
table.insert(values, repr(value))
2018-03-21 19:22:25 -07:00
]]
2018-03-21 15:03:01 -07:00
var_y = pads.stack.y + pads.stack.height
var_x = 0
--height = math.min(2+#var_names, SCREEN_H-pads.err.height-pads.stack.height)
height = SCREEN_H-(pads.err.height+pads.stack.height)
pads.vars = Pad "(V)ars", var_y,var_x,height,AUTO,var_names, ((i)=> i == @selected and color('reverse') or color())
pads.vars.on_select = (var_index)=>
2018-03-21 19:22:25 -07:00
if var_index == nil then return
2018-03-21 15:03:01 -07:00
value_x = pads.vars.x+pads.vars.width
value_w = SCREEN_W-(value_x)
2018-03-22 13:45:17 -07:00
value = values[var_index]
type_str = type(value)
2018-03-21 15:03:01 -07:00
-- Show single value:
2018-03-22 13:45:17 -07:00
switch type_str
2018-03-21 19:22:25 -07:00
when "string"
pads.values = Pad "(D)ata [string]",var_y,value_x,pads.vars.height,value_w,
2018-03-21 19:47:46 -07:00
wrap_text(value, value_w-2), (i)=>color()
2018-03-21 19:22:25 -07:00
when "table"
2018-03-22 13:45:17 -07:00
value_str = repr(value, 3)
mt = getmetatable(value)
if mt
2018-03-21 19:22:25 -07:00
type_str = if mt.__class and mt.__class.__name then mt.__class.__name
2018-03-22 13:45:17 -07:00
elseif value.__base and value.__name then "class #{value.__name}"
2018-03-21 19:22:25 -07:00
else 'table with metatable'
if mt.__tostring
value_str = tostring(value)
2018-03-22 13:45:17 -07:00
else
if value.__base and value.__name
value = value.__base
value_str = repr(value, 2)
if #value_str >= value_w-2
key_repr = (k)-> type(k) == 'string' and k or "[#{repr(k,1)}]"
value_str = table.concat ["#{key_repr k} = #{repr v,1}" for k,v in pairs(value)], "\n \n"
2018-03-21 19:22:25 -07:00
pads.values = Pad "(D)ata [#{type_str}]",var_y,value_x,pads.vars.height,value_w,
wrap_text(value_str, value_w-2), (i)=>color("cyan bold")
when "function"
info = debug.getinfo(value, 'nS')
s = ("function '%s' defined at %s:%s")\format info.name or var_names[var_index],
info.short_src, info.linedefined
2018-03-22 13:45:17 -07:00
pads.values = Pad "(D)ata [#{type_str}]",var_y,value_x,pads.vars.height,value_w,
2018-03-21 19:22:25 -07:00
wrap_text(s, value_w-2), (i)=>color("green bold")
when "number"
2018-03-22 13:45:17 -07:00
pads.values = Pad "(D)ata [#{type_str}]",var_y,value_x,pads.vars.height,value_w,
2018-03-21 19:22:25 -07:00
wrap_text(repr(value), value_w-2), (i)=>color("magenta bold")
when "boolean"
2018-03-22 13:45:17 -07:00
pads.values = Pad "(D)ata [#{type_str}]",var_y,value_x,pads.vars.height,value_w,
2018-03-21 19:22:25 -07:00
wrap_text(repr(value), value_w-2), (i)=> value and color("green bold") or color("red bold")
else
2018-03-22 13:45:17 -07:00
pads.values = Pad "(D)ata [#{type_str}]",var_y,value_x,pads.vars.height,value_w,
2018-03-21 19:22:25 -07:00
wrap_text(repr(value), value_w-2), (i)=>color()
2018-03-21 15:03:01 -07:00
collectgarbage()
collectgarbage()
pads.vars\select(1)
pads.stack.on_select = (stack_index)=>
filename,line_no = pads.stack.columns[2][stack_index]\match("([^:]*):(%d*).*")
--filename, line_no = pads.stack.lines[stack_index]\match("[^|]*| ([^:]*):(%d*).*")
line_no = tonumber(line_no)
show_src(filename, line_no)
show_vars(stack_index)
pads.stack\select(1)
selected_pad = nil
select_pad = (pad)->
if selected_pad != pad
if selected_pad
selected_pad\set_active(false)
selected_pad\refresh!
selected_pad = pad
selected_pad\set_active(true)
selected_pad\refresh!
select_pad(pads.src)
while true
for _,p in pairs(pads)
if p.dirty
p\refresh!
2018-03-21 19:47:46 -07:00
s = " press 'q' to quit "
stdscr\mvaddstr(math.floor(SCREEN_H - 1), math.floor((SCREEN_W-#s)), s)
2018-03-21 15:03:01 -07:00
C.doupdate!
c = stdscr\getch!
switch c
2018-03-21 19:30:36 -07:00
when C.KEY_DOWN, C.KEY_SR, ("j")\byte!
2018-03-21 15:03:01 -07:00
selected_pad\scroll(1,0)
when ('J')\byte!
selected_pad\scroll(10,0)
2018-03-21 19:30:36 -07:00
when C.KEY_UP, C.KEY_SF, ("k")\byte!
2018-03-21 15:03:01 -07:00
selected_pad\scroll(-1,0)
when ('K')\byte!
selected_pad\scroll(-10,0)
when C.KEY_RIGHT, ("l")\byte!
selected_pad\scroll(0,1)
when ("L")\byte!
selected_pad\scroll(0,10)
when C.KEY_LEFT, ("h")\byte!
selected_pad\scroll(0,-1)
when ("H")\byte!
selected_pad\scroll(0,-10)
when ('c')\byte!
select_pad(pads.stack) -- (C)allstack
when ('s')\byte!
select_pad(pads.src) -- (S)ource Code
when ('v')\byte!
select_pad(pads.vars) -- (V)ars
when ('d')\byte!
select_pad(pads.values) -- (D)ata
when ('o')\byte!
file = stack_locations[pads.stack.selected]
filename,line_no = file\match("([^:]*):(.*)")
line_no = tostring(pads.src.selected)
-- Launch system editor and then redraw everything
C.endwin!
os.execute((os.getenv("EDITOR") or "nano").." +"..line_no.." "..filename)
stdscr = C.initscr!
C.cbreak!
C.echo(false)
C.nl(false)
C.curs_set(0)
C.start_color!
C.use_default_colors!
stdscr\clear!
stdscr\refresh!
for _,pad in pairs(pads) do pad\refresh!
2018-03-25 15:14:18 -07:00
when C.KEY_RESIZE
stdscr\clear!
stdscr\refresh!
for _,pad in pairs(pads) do pad\refresh!
2018-03-21 15:03:01 -07:00
when ('q')\byte!, ("Q")\byte!
pads = {}
C.endwin!
return
C.endwin!
2018-03-21 14:52:55 -07:00
guard: (fn, ...)->
2018-03-21 15:03:01 -07:00
return xpcall(fn, ((err_msg)-> xpcall(ldb.run_debugger, err_hand, err_msg)), ...)
2018-03-21 14:52:55 -07:00
breakpoint: ->
2018-03-21 15:03:01 -07:00
return xpcall(ldb.run_debugger, err_hand, "Breakpoint triggered!")
2018-03-21 14:52:55 -07:00
hijack_error: ->
export error
error = (err_msg)->
2018-03-21 15:03:01 -07:00
return xpcall(ldb.run_debugger, err_hand, err_msg)
2018-03-21 14:52:55 -07:00
}
2018-03-21 15:03:01 -07:00
return ldb