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() collectgarbage() collectgarbage() local Foo = immutable({"x"}) local function countFoos() local n = 0 for h,bucket in pairs(Foo.__instances) do for _ in pairs(bucket) do n = n + 1 end end return n end local f1, f2, also_f2 = Foo(1), Foo(2), Foo(2) assert(countFoos() == 2) f1, f2 = nil, nil collectgarbage() collectgarbage() assert(countFoos() == 1) also_f2 = nil collectgarbage() collectgarbage() assert(countFoos() == 0) assert(next(Foo.__instances) == nil) 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.__index(88, 9) end)) assert(not pcall(function() return Vec.__tostring(t) end)) end) if _VERSION == "Lua 5.3" then test("Testing unpacking", function() if table.unpack then local a, b = table.unpack(Vec(5,6)) assert(a == 5 and b == 6) end end) test("Testing pairs()", function() local copy = {} for k,v in pairs(Vec(3,4)) do copy[k] = v end assert(copy.x == 3 and copy.y == 4) end) test("Testing ipairs()", function() local copy = {} for k,v in ipairs(Vec(3,4)) do copy[k] = v end assert(copy[1] == 3 and copy[2] == 4) end) end test("Testing immutable(n)", function() local Tup3 = immutable(3, {name="Tuple"}) assert(tostring(Tup3(1,2,3)) == "Tuple(1, 2, 3)") end) test("Testing tostring(class)", function() local C1 = immutable(0, {name="MYNAME"}) assert(tostring(C1) == "MYNAME") local C2 = immutable() assert(tostring(C2):match("immutable type: 0x.*")) 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