lua-immutable/tests.lua
2018-02-11 13:43:42 -08:00

189 lines
5.0 KiB
Lua

local bright = string.char(27).."[1m"
local dim = string.char(27).."[2m"
local underscore = string.char(27).."[4m"
local red = string.char(27).."[31m"
local green = string.char(27).."[32m"
local reset = string.char(27).."[0m"
print(bright..underscore.."\nTesting with ".._VERSION..":"..reset)
repr = function(t)
local buff = tostring(t)
if type(t) == 'table' then
buff = "{"..buff.." "
for k, v in pairs(t) do
buff = buff..("%s = %s, "):format(k,v)
end
buff = buff.."}"
end
return buff
end
local num_errors = 0
local test_number = 0
local function test(description, fn)
test_number = test_number + 1
description = (bright.."% 3d. "..reset.."%s"):format(test_number, description)
io.write(description)
io.write(red)
local ok, err = pcall(fn)
if not ok then
io.write(reset..dim.."\r..................................")
io.write(reset.."["..bright..red.."FAILED"..reset.."]\r")
io.write("\r"..description.."\n"..reset)
print(reset..red..(err or "")..reset)
num_errors = num_errors + 1
else
io.write(reset..dim.."\r..................................")
io.write(reset.."["..green.."PASSED"..reset.."]\r")
io.write(description.."\n")
end
end
test("Loading module", function()
immutable = require"immutable"
end)
test("Creating class", function()
Vec = immutable({"x","y"}, {
len2=function(self)
return self.x*self.x + self.y*self.y
end,
__add=function(self, other)
local cls = getmetatable(self)
return cls(self.x+other.x, self.y+other.y)
end,
__tostring=function(self)
return "Vec("..tostring(self.x)..", "..tostring(self.y)..")"
end,
})
end)
test("Instantiating class", function()
v = assert(Vec(1,3))
end)
test("Testing # operator", function()
assert(#v == 2)
end)
test("Testing method", function()
assert(v:len2() == 10)
end)
test("Testing tostring", function()
assert(tostring(v) == "Vec(1, 3)")
end)
test("Testing from_table", function()
assert(Vec:from_table(setmetatable({y=3}, {__index={x=1}})) == v)
assert(Vec:from_table({x=1}) == Vec(1, nil))
end)
test("Testing equality", function()
also_v = Vec(1,3)
assert(v == also_v)
assert(tostring(v) == tostring(also_v))
-- Hash collision in the current implementation
not_v = Vec(true,3)
assert(v ~= not_v)
assert(tostring(not_v) == "Vec(true, 3)")
also_not_v = Vec(true,3)
assert(not_v == also_not_v)
end)
test("Testing singletons", function()
local T1 = immutable()
local T2 = immutable()
assert(T1() == T1())
assert(T1() ~= T2())
end)
test("Testing __add metamethod", function()
assert(v + Vec(5,6) == Vec(6,9))
end)
collectgarbage()
test("Testing string members", function()
Vec("hello", "world")
end)
test("Testing table members", function()
Vec({}, {})
end)
test("Testing function members", function()
Vec(function() end, function() end)
end)
test("Testing immutable members", function()
Vec(v, v)
end)
collectgarbage()
collectgarbage()
test("Testing garbage collection", function()
local collected = false
local GCSnooper = immutable({}, {
__gc=function(self)
collected = true
end,
})
local g = GCSnooper()
collectgarbage()
collectgarbage()
assert(not collected)
g = nil
collectgarbage()
collectgarbage()
assert(collected)
end)
test("Testing stupid metamethods", function()
local Five = immutable({"x"}, {__index=function() return 5 end, derp = 99})
local f = Five(99)
assert(f.x == 5 and f.asdf == 5 and f.derp == 5)
end)
test("Testing similar class", function()
FooVec = immutable({"x","y"}, {
len2=function(self)
return self.x*self.x + self.y*self.y
end,
__add=function(self, other)
local cls = getmetatable(self)
return cls(self.x+other.x, self.y+other.y)
end,
__tostring=function(self)
return "Vec("..tostring(self.x)..", "..tostring(self.y)..")"
end,
})
assert(FooVec(1,1) ~= Vec(1,1))
end)
test("Testing is_instance", function()
fv = FooVec(1,2)
assert(FooVec:is_instance(fv))
assert(not FooVec:is_instance(v))
assert(not FooVec:is_instance("asdf"))
assert(not FooVec:is_instance({}))
assert(not FooVec:is_instance(5))
end)
test("Testing spoofing", function()
local t = {99,100}
setmetatable(t, Vec)
assert(not pcall(function() return t.x end))
assert(not pcall(function() return Vec(88) end))
assert(not pcall(function() return Vec.__tostring(t) end))
end)
test("Testing unpacking", function()
if table.unpack then
local a, b = table.unpack(Vec(5,6))
assert(a == 5 and b == 6)
end
end)
if num_errors == 0 then
print(green.."All tests passed!"..reset)
else
print(bright..red.."*** "..tostring(num_errors).." test"..(num_errors > 1 and "s" or "").." failed! ***"..reset)
io.write(reset)
os.exit(false, true)
end