1
0
pretty/test/test_resilience.lua

218 lines
7.6 KiB
Lua

--------------------------------------------------------------------------------
-- Lua 5.2 compat
if not setfenv then -- Lua 5.2
-- based on http://lua-users.org/lists/lua-l/2010-06/msg00314.html
-- this assumes f is a function
local function findenv(f)
local level = 1
repeat
local name, value = debug.getupvalue(f, level)
if name == '_ENV' then return level, value end
level = level + 1
until name == nil
return nil end
getfenv = function (f) return(select(2, findenv(f)) or _G) end
setfenv = function (f, t)
local level = findenv(f)
if level then debug.setupvalue(f, level, t) end
return f end
end
--------------------------------------------------------------------------------
local SUITE = require 'TestSuite' 'resilience'
SUITE:setEnvironment{
pretty = require('pretty'),
setfenv = setfenv,
}
--------------------------------------------------------------------------------
-- Prevent usage of unknown options.
SUITE:addTest('Dont allow unknown options', function ()
local error_msg = bad_call(pretty, 'Hello World', { some_random_option = true })
assert(error_msg)
end)
SUITE:addTest('Dont allow bad types in options', function ()
local error_msg = bad_call(pretty, 'Hello World', { indent = 51 })
assert(error_msg)
end)
--------------------------------------------------------------------------------
-- Strange standard libs
SUITE:addTest('no_std_lib', function ()
-- This tests whether one could load the library with an empty env, without
-- an error.
local env = {}
local library = setfenv(loadfile('./library.lua'), env)()
for func, func_info in pairs(library) do
error(('For some reason %s is defined in the library'):format(func_info.name))
end
end)
SUITE:addTest('a_very_small_part_of_math', function ()
-- This tests whether one could load the library with an empty env, without
-- an error.
local env = { debug = debug, math = { abs = math.abs } }
local library = setfenv(loadfile('./library.lua'), env)()
assert( library[math.abs], 'Why is math.abs not defined in the library?' )
end)
SUITE:addTest('redefined part of math', function ()
-- This tests whether the library ignores a redefined function of an
-- builtin, when defining documentation.
local env = { debug = debug, setfenv = setfenv, loadfile = function() end }
local library = setfenv(loadfile('./library.lua'), env)()
assert( library[env.setfenv], 'Why is setfenv not defined in the library?' )
assert( not library[env.loadfile], 'Why is loadfile defined in the library?' )
end)
SUITE:addTest('redefined part of math 2', function ()
-- This tests whether the library allows a redefined function of an
-- builtin, when defining documentation, due to not being able to determine
-- the origin of the function.
local env = { setfenv = setfenv, loadfile = function() end }
local library = setfenv(loadfile('./library.lua'), env)()
assert( library[env.setfenv], 'Why is setfenv not defined in the library?' )
assert( library[env.loadfile], 'Why is loadfile not defined in the library?' )
end)
--------------------------------------------------------------------------------
if false then
-- This test takes a long time.
local data = {}
for str, name in SUITE:getExamples 'STRING_UTF8_MALFORMED' do
data[#data+1] = {str, name}
end
local pjk_path = '/tmp/test_pjk_'..os.time()
local conf = [[
function love.conf(t)
-- Graphics Settings
t.window.width = 1
t.window.height = 1
t.window.minwidth = 1
t.window.minheight = 1
-- Modules
t.modules.audio = false
t.modules.event = true
t.modules.graphics = true
t.modules.image = false
t.modules.joystick = false
t.modules.keyboard = false
t.modules.math = false
t.modules.mouse = false
t.modules.physics = false
t.modules.sound = false
t.modules.system = false
t.modules.timer = false
t.modules.touch = false
t.modules.video = false
t.modules.window = true
t.modules.thread = false
end]]
-- Create project, main.lua and conf.lua
os.execute('mkdir '..pjk_path)
local f = io.open(pjk_path..'/conf.lua', 'w')
f:write(conf)
f:close()
local f = io.open(pjk_path..'/main.lua', 'w')
f:write''
f:close()
SUITE:addTest('Proper malformed utf8 escaping (through LÖVE)', function (str, name)
-- This test asserts that malformed unicode strings are escaped. This is
-- done by checking whether LÖVE can draw the sequence of bytes, given by
-- `pretty`. LÖVE have robust unicode checking, and will throw an error if
-- the characters is not escaped properly.
-- The input strings are gotten from TestSuite's example strings.
--do return error 'Test skipped' end
-- Create project and main.lua
local f = io.open(pjk_path..'/main.lua', 'w')
f:write('love.errhand = love.event.quit love.graphics.print('..pretty(str)..') love.event.quit(0)')
f:close()
-- Run project
local status_code = os.execute('love '..pjk_path)
if status_code ~= 0 then
local l = {}
for i = 1, #str do l[i] = tostring(str:byte(i)) end
error(('Could not escape the string "%s" properly: %s\n\tByte-sequence: \\%s'):format(name, str, table.concat(l, '\\')))
end
end, { data = data })
end
--------------------------------------------------------------------------------
-- Strange metatable
SUITE:addTest('Metatable: Error on operator', function (op)
pretty( setmetatable( { a = 2, b = 4, 1, 2, 3 } , {[op] = function() error 'This should be unreachable!' end}) )
assert(true)
end, { data = { '__add', '__sub', '__mul', '__div', '__mod', '__pow', '__unm', '__concat','__len', '__eq', '__lt', '__le', '__index', '__newindex', '__call', '__tostring' } })
SUITE:addTest('Metatable: # returns bogus length', function ()
assert_equal('{ 1, 2, 3, 4 }', pretty( setmetatable({ 1, 2, 3, 4 }, {__len = function() return 2 end})))
end)
SUITE:addTest('Metatable: object is never equal to anything, not even itself', function ()
assert_equal('{ 1, 2, 3, 4 }', pretty( setmetatable({ 1, 2, 3, 4 }, {__eq = function() return false end})))
end)
SUITE:addTest('Metatable: object contains nothing, but has prototype', function ()
pretty( setmetatable({}, {__index = { a = 2, b = 5, c = 2 }}) )
assert(true)
-- TODO: Determine what to output do when this happens.
end)
SUITE:addTest('Metatable: object contains every odd number', function ()
pretty( setmetatable({}, {__index = function(t, i) return i*2-1 end}) )
assert(true)
-- TODO: Determine what to output do when this happens.
end)
SUITE:addTest('Metatable: object is also callable', function ()
pretty( setmetatable({}, {__call = function(t, i) return i*2-1 end}) )
assert(true)
-- TODO: Determine what to output do when this happens.
end)
--------------------------------------------------------------------------------
-- Metatable __tostring
do
local data = {}
for str, name in SUITE:getExamples 'TABLE_METAMETHODS' do
data[#data+1] = { name, str }
end
SUITE:addTest('Metatable shenanigans', function (_, str)
pretty(str)
assert(true)
end, { data = data })
end
-- TODO: Tests for alternative modes (__mode = 'k', __mode = 'v', __mode = 'kv')
-- TODO: Tests for __metatable
--------------------------------------------------------------------------------
return SUITE