2019-01-11 18:46:12 +02:00
|
|
|
-- localised functions
|
2018-11-24 21:44:02 +02:00
|
|
|
local format = string.format
|
2019-01-11 18:46:12 +02:00
|
|
|
local match = string.match
|
|
|
|
local gsub = string.gsub
|
2018-11-24 21:44:02 +02:00
|
|
|
local serialize = serpent.line
|
2018-12-28 18:21:45 +02:00
|
|
|
local debug_getupvalue = debug.getupvalue
|
2018-11-24 21:44:02 +02:00
|
|
|
|
|
|
|
-- this
|
|
|
|
local Debug = {}
|
|
|
|
|
|
|
|
global.debug_message_count = 0
|
|
|
|
|
|
|
|
---@return number next index
|
|
|
|
local function increment()
|
|
|
|
local next = global.debug_message_count + 1
|
|
|
|
global.debug_message_count = next
|
|
|
|
|
|
|
|
return next
|
|
|
|
end
|
|
|
|
|
2019-01-11 18:46:12 +02:00
|
|
|
--- Takes the table output from debug.getinfo and pretties it
|
2019-01-03 22:04:09 +02:00
|
|
|
local function cleanup_debug(debug_table)
|
2019-01-11 18:46:12 +02:00
|
|
|
local short_src = match(debug_table.source, '/[^/]*/[^/]*$')
|
|
|
|
-- require will not return a valid string so short_src may be nil here
|
|
|
|
if short_src then
|
|
|
|
short_src = gsub(short_src, '%.lua', '')
|
|
|
|
end
|
|
|
|
|
|
|
|
return format('[function: %s file: %s line number: %s]', debug_table.name, short_src, debug_table.currentline)
|
2019-01-03 22:04:09 +02:00
|
|
|
end
|
|
|
|
|
2018-11-24 21:44:02 +02:00
|
|
|
---Shows the given message if debug is enabled. Uses serpent to print non scalars.
|
2019-01-03 22:04:09 +02:00
|
|
|
-- @param message <table|string|number|boolean>
|
2019-01-11 18:46:12 +02:00
|
|
|
-- @param stack_traceback <number|nil> levels of stack trace to give, defaults to 1 level if nil
|
|
|
|
function Debug.print(message, trace_levels)
|
2018-11-24 21:44:02 +02:00
|
|
|
if not _DEBUG then
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
2019-01-11 18:46:12 +02:00
|
|
|
if not trace_levels then
|
|
|
|
trace_levels = 2
|
|
|
|
else
|
|
|
|
trace_levels = trace_levels + 1
|
|
|
|
end
|
|
|
|
|
|
|
|
local traceback_string = ''
|
2018-11-24 21:44:02 +02:00
|
|
|
if type(message) ~= 'string' and type(message) ~= 'number' and type(message) ~= 'boolean' then
|
|
|
|
message = serialize(message)
|
|
|
|
end
|
|
|
|
|
|
|
|
message = format('[%d] %s', increment(), tostring(message))
|
2019-01-11 18:46:12 +02:00
|
|
|
|
|
|
|
if trace_levels >= 2 then
|
|
|
|
for i = 2, trace_levels do
|
|
|
|
local debug_table = debug.getinfo(i)
|
|
|
|
if debug_table then
|
|
|
|
traceback_string = format('%s -> %s', traceback_string, cleanup_debug(debug_table))
|
|
|
|
else
|
|
|
|
break
|
|
|
|
end
|
|
|
|
end
|
|
|
|
message = format('%s - Traceback%s', message, traceback_string)
|
2019-01-03 22:04:09 +02:00
|
|
|
end
|
2019-01-11 18:46:12 +02:00
|
|
|
|
2019-01-03 22:04:09 +02:00
|
|
|
if game then
|
|
|
|
game.print(message)
|
|
|
|
end
|
2018-11-24 21:44:02 +02:00
|
|
|
log(message)
|
|
|
|
end
|
|
|
|
|
|
|
|
---Shows the given message if debug is on.
|
|
|
|
---@param position Position
|
|
|
|
---@param message string
|
|
|
|
function Debug.print_position(position, message)
|
|
|
|
Debug.print(format('%s %s', serialize(position), message))
|
|
|
|
end
|
|
|
|
|
|
|
|
---Executes the given callback if cheating is enabled.
|
|
|
|
---@param callback function
|
|
|
|
function Debug.cheat(callback)
|
|
|
|
if _CHEATS then
|
|
|
|
callback()
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-12-28 18:21:45 +02:00
|
|
|
--- Returns true if the function is a closure, false otherwise.
|
|
|
|
-- A closure is a function that contains 'upvalues' or in other words
|
|
|
|
-- has a reference to a local variable defined outside the function's scope.
|
|
|
|
-- @param func<function>
|
|
|
|
-- @return boolean
|
|
|
|
function Debug.is_closure(func)
|
|
|
|
local i = 1
|
|
|
|
while true do
|
|
|
|
local n = debug_getupvalue(func, i)
|
|
|
|
|
|
|
|
if n == nil then
|
|
|
|
return false
|
|
|
|
elseif n ~= '_ENV' then
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
|
|
|
|
i = i + 1
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-11-24 21:44:02 +02:00
|
|
|
return Debug
|