lua-immutable/tests.lua
2018-06-04 18:35:11 -07:00

378 lines
11 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
--os.exit(true)
else
io.write(reset..dim.."\r.......................................")
io.write(reset.."["..green.."PASSED"..reset.."]\r")
io.write(description.."\n")
end
end
test("Loading module", function()
package.cpath = "./immutable.so;"..package.cpath
immutable = require"immutable"
end)
test("Creating class", function()
Vec = immutable({"x","y"}, {
name="Vec",
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,
})
end)
test("Instantiating class", function()
v = assert(Vec(1,3))
end)
test("Testing indexing", function()
assert(v.x == 1)
end)
test("Testing # operator", function()
assert(#v == 0)
local T = immutable()
assert(#T(1,2,3) == 3)
end)
test("Testing # operator for mixed table", function()
assert(#Vec(1,2,3,4,5) == 3)
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({x=1,y=2}) == Vec(1,2), "assertion 1 failed!")
assert(Vec:from_table({3, x=1,y=2}) == Vec(1,2,3), "assertion 2 failed!")
assert(Vec:from_table(setmetatable({y=3}, {__index={x=1}})) == v, "assertion 3 failed!")
assert(Vec:from_table({x=1}) == Vec(1, nil), "assertion 4 failed!")
end)
test("Testing from_table for tuples", function()
local T = immutable(nil)
assert(T(1,2) == T:from_table({1,2,n=2}))
assert(T(1,2) == T:from_table({1,2}))
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 Foo = immutable({"x"})
local function countFoos()
collectgarbage()
collectgarbage()
local n = 0
local buckets = 0
for h,bucket in pairs(Foo.__instances) do
buckets = buckets + 1
for _ in pairs(bucket) do
n = n + 1
end
end
collectgarbage()
collectgarbage()
return n, buckets
end
local f1, f2, also_f2 = Foo(1), Foo(2), Foo(2)
local foos, buckets = countFoos()
assert(foos == 2, "WTF? "..tostring(foos))
f1, f2 = nil, nil
foos, buckets = countFoos()
assert(foos == 1)
also_f2 = nil
foos, buckets = countFoos()
assert(foos == 0, "Leaking instances")
assert(buckets == 0, "Leaking hash buckets")
assert(next(Foo.__instances) == nil)
end)
test("Testing __index", function()
local Baz = immutable({"x","y"}, {z=99})
local b = Baz(1,nil)
assert(b.x == 1 and b.y == nil and b.z == 99 and b.asdf == nil)
local Foo = immutable({"x","y"}, {z=99, __index=function() return "foo" end})
local f = Foo(1,nil)
assert(f.x == "foo" and f.y == "foo" and f.z == "foo" and f.asdf == "foo")
end)
test("Testing __rawindex", function()
local Foo = immutable({"x","y"}, {
classvar = 99,
__index=function(self,k)
local cls = getmetatable(self)
local inst_val, found = cls.__super.__index(self, k)
if found then return inst_val end
return cls[k] or tostring(k)..tostring(cls.__super.__index(self, 'x'))
end,
})
local f = Foo(23, nil)
assert(f.x == 23 and f.y == nil and f.classvar == 99 and f.asdf == "asdf23")
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.__index(88, 9) end))
assert(not pcall(function() return Vec.__tostring(t) end))
end)
if _VERSION == "Lua 5.3" then
test("Testing unpacking", function()
local Tuple = immutable()
if table.unpack then
local a, b = table.unpack(Tuple(5,6))
assert(a == 5 and b == 6)
end
end)
end
test("Testing immutable(nil)", function()
local Tup3 = immutable(nil, {name="Tuple"})
assert(tostring(Tup3(1,2,3)) == "Tuple(1, 2, 3)")
end)
test("Testing tostring(class)", function()
local C1 = immutable(nil, {name="MYNAME"})
assert(tostring(C1) == "MYNAME")
local C2 = immutable()
assert(tostring(C2):match("immutable type: 0x.*"))
end)
test("Testing tuple tostring", function()
local tup3 = immutable(nil)
assert(tostring(tup3(1,2,3)) == "(1, 2, 3)")
assert(tostring(tup3(1,tup3(2,3,4),5)) == "(1, (2, 3, 4), 5)")
end)
test("Testing giant immutable table", function()
local keys = {}
local N = 100000
for i=1,N do keys[i] = "key_"..tostring(i) end
local T = immutable(keys)
local values = {}
for i,key in ipairs(keys) do values[key] = i*i end
pcall(function() T:from_table(values) end)
end)
if _VERSION == "Lua 5.2" or _VERSION == "Lua 5.3" then
test("Testing tuple iteration", function()
local T = immutable()
local t = T(1,4,9,nil,16,nil)
local checks = {1,4,9,nil,16,nil}
local passed, iterations = 0, 0
for i,v in ipairs(t) do
assert(checks[i] == v)
passed = passed + 1
iterations = iterations + 1
end
assert(passed == 6 and iterations == passed)
passed, iterations = 0, 0
for k,v in pairs(t) do
assert(checks[k] == v)
passed = passed + 1
iterations = iterations + 1
end
assert(passed == 6 and iterations == passed)
end)
test("Testing table iteration", function()
local Foo = immutable({"x", "y", "z","w"})
local f = Foo(1,nil,2,nil)
local checks = {1,nil,2,nil}
local passed, iterations = 0, 0
checks = {x=1,y=nil,z=2,w=nil}
for k,v in pairs(f) do
assert(checks[k] == v)
passed = passed + 1
iterations = iterations + 1
end
assert(passed == 4 and iterations == passed)
end)
test("Testing mixed iteration", function()
local T = immutable({"skip"})
local t = T('a','b','c')
local checks = {'b','c'}
local passed, iterations = 0, 0
for i,v in ipairs(t) do
assert(checks[i] == v, "checks["..i.."] ~= "..v)
passed = passed + 1
iterations = iterations + 1
end
assert(passed == 2 and iterations == passed, "ipairs failed")
checks = {skip='a','b','c'}
passed = 0
for k,v in pairs(t) do
assert(checks[k] == v, "checks["..k.."] ~= "..v)
passed = passed + 1
end
assert(passed == 3, "pairs failed")
end)
end
test("Testing __new", function()
local Foo = immutable({"x","y"}, {
__new=function(cls,x,y)
return x+1,y+1
end
})
local f = Foo(2,3)
assert(f.x == 3 and f.y == 4)
end)
test("Testing __new with #args", function()
local Foo = immutable({"x","y"}, {
__new=function(cls,x,y,z)
return x+z,y+z
end
})
local f = Foo(2,3,1)
assert(f.x == 3 and f.y == 4)
end)
test("Testing __new with varargs", function()
local Foo = immutable(nil, {
__new=function(cls,...)
return "prefix",...
end
})
local f = Foo(1,2)
assert(f[1] == "prefix" and f[2] == 1 and f[3] == 2 and f[4] == nil)
end)
test("Testing __hash", function()
local T = immutable()
local t = T(1,2,3)
local h = t:__hash()
assert(type(h) == 'number')
assert(T.__instances[h][t])
end)
test("Testing __rawindex", function()
local T = immutable({"x"}, {
classvar = 99,
__index = function(self, key)
local cls = getmetatable(self)
local value, found = cls.__super.__index(self, key)
if not found then
return cls[key]
else
return tostring(value)
end
end
})
assert(T(3).x == "3" and T(nil).x == "nil" and T(1).classvar == 99)
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)
error()
end