295 lines
8.0 KiB
Lua
295 lines
8.0 KiB
Lua
|
package.path = string.format("../lua/?.lua;./?.lua;%s",package.path)
|
||
|
|
||
|
local function checkReadBuffer(buf, offset, sizePrefix)
|
||
|
offset = offset or 0
|
||
|
|
||
|
if type(buf) == "string" then
|
||
|
buf = flatbuffers.binaryArray.New(buf)
|
||
|
end
|
||
|
|
||
|
if sizePrefix then
|
||
|
local size = flatbuffers.N.Int32:Unpack(buf, offset)
|
||
|
-- no longer matches python tests, but the latest 'monsterdata_test.mon'
|
||
|
-- is 448 bytes, minus 4 to arrive at the 444
|
||
|
assert(size == 444)
|
||
|
offset = offset + flatbuffers.N.Int32.bytewidth
|
||
|
end
|
||
|
|
||
|
local mon = monster.GetRootAsMonster(buf, offset)
|
||
|
assert(mon:Hp() == 80, "Monster Hp is not 80")
|
||
|
assert(mon:Mana() == 150, "Monster Mana is not 150")
|
||
|
assert(mon:Name() == "MyMonster", "Monster Name is not MyMonster")
|
||
|
|
||
|
local vec = assert(mon:Pos(), "Monster Position is nil")
|
||
|
assert(vec:X() == 1.0)
|
||
|
assert(vec:Y() == 2.0)
|
||
|
assert(vec:Z() == 3.0)
|
||
|
assert(vec:Test1() == 3.0)
|
||
|
assert(vec:Test2() == 2)
|
||
|
|
||
|
local t = require("MyGame.Example.Test").New()
|
||
|
t = assert(vec:Test3(t))
|
||
|
|
||
|
assert(t:A() == 5)
|
||
|
assert(t:B() == 6)
|
||
|
|
||
|
local ut = require("MyGame.Example.Any")
|
||
|
assert(mon:TestType() == ut.Monster)
|
||
|
|
||
|
local table2 = mon:Test()
|
||
|
assert(getmetatable(table2) == "flatbuffers.view.mt")
|
||
|
|
||
|
local mon2 = monster.New()
|
||
|
mon2:Init(table2.bytes, table2.pos)
|
||
|
|
||
|
assert(mon2:Name() == "Fred")
|
||
|
|
||
|
assert(mon:InventoryLength() == 5)
|
||
|
local invsum = 0
|
||
|
for i=1,mon:InventoryLength() do
|
||
|
local v = mon:Inventory(i)
|
||
|
invsum = invsum + v
|
||
|
end
|
||
|
assert(invsum == 10)
|
||
|
|
||
|
for i=1,5 do
|
||
|
assert(mon:VectorOfLongs(i) == 10^((i-1)*2))
|
||
|
end
|
||
|
|
||
|
local dbls = { -1.7976931348623157e+308, 0, 1.7976931348623157e+308}
|
||
|
for i=1,mon:VectorOfDoublesLength() do
|
||
|
assert(mon:VectorOfDoubles(i) == dbls[i])
|
||
|
end
|
||
|
|
||
|
assert(mon:Test4Length() == 2)
|
||
|
|
||
|
local test0 = mon:Test4(1)
|
||
|
local test1 = mon:Test4(2)
|
||
|
|
||
|
local v0 = test0:A()
|
||
|
local v1 = test0:B()
|
||
|
local v2 = test1:A()
|
||
|
local v3 = test1:B()
|
||
|
|
||
|
local sumtest12 = v0 + v1 + v2 + v3
|
||
|
assert(sumtest12 == 100)
|
||
|
|
||
|
assert(mon:TestarrayofstringLength() == 2)
|
||
|
assert(mon:Testarrayofstring(1) == "test1")
|
||
|
assert(mon:Testarrayofstring(2) == "test2")
|
||
|
|
||
|
assert(mon:TestarrayoftablesLength() == 0)
|
||
|
assert(mon:TestnestedflatbufferLength() == 0)
|
||
|
assert(mon:Testempty() == nil)
|
||
|
end
|
||
|
|
||
|
local function generateMonster(sizePrefix)
|
||
|
local b = flatbuffers.Builder(0)
|
||
|
local str = b:CreateString("MyMonster")
|
||
|
local test1 = b:CreateString("test1")
|
||
|
local test2 = b:CreateString("test2")
|
||
|
local fred = b:CreateString("Fred")
|
||
|
|
||
|
monster.StartInventoryVector(b, 5)
|
||
|
b:PrependByte(4)
|
||
|
b:PrependByte(3)
|
||
|
b:PrependByte(2)
|
||
|
b:PrependByte(1)
|
||
|
b:PrependByte(0)
|
||
|
local inv = b:EndVector(5)
|
||
|
|
||
|
monster.Start(b)
|
||
|
monster.AddName(b, fred)
|
||
|
local mon2 = monster.End(b)
|
||
|
|
||
|
monster.StartTest4Vector(b, 2)
|
||
|
test.CreateTest(b, 10, 20)
|
||
|
test.CreateTest(b, 30, 40)
|
||
|
local test4 = b:EndVector(2)
|
||
|
|
||
|
monster.StartTestarrayofstringVector(b, 2)
|
||
|
b:PrependUOffsetTRelative(test2)
|
||
|
b:PrependUOffsetTRelative(test1)
|
||
|
local testArrayOfString = b:EndVector(2)
|
||
|
|
||
|
monster.StartVectorOfLongsVector(b, 5)
|
||
|
b:PrependInt64(100000000)
|
||
|
b:PrependInt64(1000000)
|
||
|
b:PrependInt64(10000)
|
||
|
b:PrependInt64(100)
|
||
|
b:PrependInt64(1)
|
||
|
local vectorOfLongs = b:EndVector(5)
|
||
|
|
||
|
monster.StartVectorOfDoublesVector(b, 3)
|
||
|
b:PrependFloat64(1.7976931348623157e+308)
|
||
|
b:PrependFloat64(0)
|
||
|
b:PrependFloat64(-1.7976931348623157e+308)
|
||
|
local vectorOfDoubles = b:EndVector(3)
|
||
|
|
||
|
monster.Start(b)
|
||
|
local pos = vec3.CreateVec3(b, 1.0, 2.0, 3.0, 3.0, 2, 5, 6)
|
||
|
monster.AddPos(b, pos)
|
||
|
|
||
|
monster.AddHp(b, 80)
|
||
|
monster.AddName(b, str)
|
||
|
monster.AddInventory(b, inv)
|
||
|
monster.AddTestType(b, 1)
|
||
|
monster.AddTest(b, mon2)
|
||
|
monster.AddTest4(b, test4)
|
||
|
monster.AddTestarrayofstring(b, testArrayOfString)
|
||
|
monster.AddVectorOfLongs(b, vectorOfLongs)
|
||
|
monster.AddVectorOfDoubles(b, vectorOfDoubles)
|
||
|
local mon = monster.End(b)
|
||
|
|
||
|
if sizePrefix then
|
||
|
b:FinishSizePrefixed(mon)
|
||
|
else
|
||
|
b:Finish(mon)
|
||
|
end
|
||
|
return b:Output(true), b:Head()
|
||
|
end
|
||
|
|
||
|
local function sizePrefix(sizePrefix)
|
||
|
local buf,offset = generateMonster(sizePrefix)
|
||
|
checkReadBuffer(buf, offset, sizePrefix)
|
||
|
end
|
||
|
|
||
|
local function testCanonicalData()
|
||
|
local f = assert(io.open('monsterdata_test.mon', 'rb'))
|
||
|
local wireData = f:read("*a")
|
||
|
f:close()
|
||
|
checkReadBuffer(wireData)
|
||
|
end
|
||
|
|
||
|
local function benchmarkMakeMonster(count)
|
||
|
local length = #(generateMonster())
|
||
|
|
||
|
--require("flatbuffers.profiler")
|
||
|
--profiler = newProfiler("call")
|
||
|
--profiler:start()
|
||
|
|
||
|
local s = os.clock()
|
||
|
for i=1,count do
|
||
|
generateMonster()
|
||
|
end
|
||
|
local e = os.clock()
|
||
|
|
||
|
--profiler:stop()
|
||
|
|
||
|
--local outfile = io.open( "profile.txt", "w+" )
|
||
|
--profiler:report( outfile, true)
|
||
|
--outfile:close()
|
||
|
|
||
|
|
||
|
local dur = (e - s)
|
||
|
local rate = count / (dur * 1000)
|
||
|
local data = (length * count) / (1024 * 1024)
|
||
|
local dataRate = data / dur
|
||
|
|
||
|
print(string.format('built %d %d-byte flatbuffers in %.2fsec: %.2f/msec, %.2fMB/sec',
|
||
|
count, length, dur, rate, dataRate))
|
||
|
end
|
||
|
|
||
|
local function benchmarkReadBuffer(count)
|
||
|
local f = assert(io.open('monsterdata_test.mon', 'rb'))
|
||
|
local buf = f:read("*a")
|
||
|
f:close()
|
||
|
|
||
|
local s = os.clock()
|
||
|
for i=1,count do
|
||
|
checkReadBuffer(buf)
|
||
|
end
|
||
|
local e = os.clock()
|
||
|
|
||
|
local dur = (e - s)
|
||
|
local rate = count / (dur * 1000)
|
||
|
local data = (#buf * count) / (1024 * 1024)
|
||
|
local dataRate = data / dur
|
||
|
|
||
|
print(string.format('traversed %d %d-byte flatbuffers in %.2fsec: %.2f/msec, %.2fMB/sec',
|
||
|
count, #buf, dur, rate, dataRate))
|
||
|
end
|
||
|
|
||
|
local tests =
|
||
|
{
|
||
|
{
|
||
|
f = sizePrefix,
|
||
|
d = "Test size prefix",
|
||
|
args = {{true}, {false}}
|
||
|
},
|
||
|
{
|
||
|
f = testCanonicalData,
|
||
|
d = "Tests Canonical flatbuffer file included in repo"
|
||
|
},
|
||
|
{
|
||
|
f = benchmarkMakeMonster,
|
||
|
d = "Benchmark making monsters",
|
||
|
args = {
|
||
|
{100},
|
||
|
{1000},
|
||
|
{10000},
|
||
|
}
|
||
|
},
|
||
|
{
|
||
|
f = benchmarkReadBuffer,
|
||
|
d = "Benchmark reading monsters",
|
||
|
args = {
|
||
|
{100},
|
||
|
{1000},
|
||
|
{10000},
|
||
|
-- uncomment following to run 1 million to compare.
|
||
|
-- Took ~141 seconds on my machine
|
||
|
--{1000000},
|
||
|
}
|
||
|
},
|
||
|
}
|
||
|
|
||
|
local result, err = xpcall(function()
|
||
|
flatbuffers = assert(require("flatbuffers"))
|
||
|
monster = assert(require("MyGame.Example.Monster"))
|
||
|
test = assert(require("MyGame.Example.Test"))
|
||
|
vec3 = assert(require("MyGame.Example.Vec3"))
|
||
|
|
||
|
local function buildArgList(tbl)
|
||
|
local s = ""
|
||
|
for _,item in ipairs(tbl) do
|
||
|
s = s .. tostring(item) .. ","
|
||
|
end
|
||
|
return s:sub(1,-2)
|
||
|
end
|
||
|
|
||
|
local testsPassed, testsFailed = 0,0
|
||
|
for _,test in ipairs(tests) do
|
||
|
local allargs = test.args or {{}}
|
||
|
for _,args in ipairs(allargs) do
|
||
|
local results, err = xpcall(test.f,debug.traceback, table.unpack(args))
|
||
|
if results then
|
||
|
testsPassed = testsPassed + 1
|
||
|
else
|
||
|
testsFailed = testsFailed + 1
|
||
|
print(string.format(" Test [%s](%s) failed: \n\t%s",
|
||
|
test.d or "",
|
||
|
buildArgList(args),
|
||
|
err))
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
local totalTests = testsPassed + testsFailed
|
||
|
print(string.format("# of test passed: %d / %d (%.2f%%)",
|
||
|
testsPassed,
|
||
|
totalTests,
|
||
|
totalTests ~= 0
|
||
|
and 100 * (testsPassed / totalTests)
|
||
|
or 0)
|
||
|
)
|
||
|
|
||
|
return 0
|
||
|
end, debug.traceback)
|
||
|
|
||
|
if not result then
|
||
|
print("Unable to run tests due to test framework error: ",err)
|
||
|
end
|
||
|
|
||
|
os.exit(result or -1)
|